LINUX系统编程 由REDIS的持久化机制联想到的子进程退出的相关问题
2014-08-27 22:53
218 查看
19:22:01 2014-08-27
引言:
以前对wait waitpid 以及exit这几个函数只是大致上了解,但是看REDIS的AOF和RDB 2种持久化时 均要处理子进程运行完成退出和父进程需要做的什么事情,所以特定看了UNIX环境编程和LINUX系统编程这2本书 重新梳理下整个要点。
内容:
一般而言: 如果程序类似于下面的情况:
子进程就会调用了我们进程退出系统调用exit()分为2种状态 0: 成功 非0:失败
exit需要做以下事情:
1: 关闭所有打开的IO流
2 删除所有的临时文件
1 2是传说中用户态需要完成的事情 接下来就调用_exit() 【这个函数在用户态下 是被exit()封装起来的函数】触发内核来完成进程退出的相关工作
接下来就是清空所有的子进程所需要的资源哦 最后完成了 就会发送一个进程退出状态【信号】给父进程 父进程做什么处理就是编程人员完成的。
子进程发送的信号是SIGCHLD 一般而言就像REDIS fork()了一个进程之后 子进程完成了相应的持久化动作之后,父进程需要了解子进程一些状态来做相应的处理工作,所以在LINUX系统中,这个子进程退出后 还有一个标记位 我们称作zombie进程。父进程必须来取子进程的相应zombie的相应标记信息 不然是不会消除的。
所以对于父进程而言,必须调用相应的函数【首先是注册了SIGCHLD】处理这些标记位哦 wait/waitpid就是干这种事情。我们来看看这系列的调用所要做的事情:
wait等待子进程挂 结果可以读出子进程是被信号量终止了 所以说 读出子进程的状态还是非常有实际意义的~!
接下来看看wait/waitpid函数的区别:
如果有多个子进程 我该怎么办 要么选择多个wait 但是同时到达 可能就会处理一个子进程 而其他进程状态没有被处理 造成了很多zombie进程 这是不行的 这样就有了waitpid函数:
waitpid(pid_t pid,int *status,int options)
pid=-1 等待任意一子进程
>0 要传入的那个子进程
参数:
WNOHANG: 不会阻塞 立刻返回。 其他参数暂时不去研究了
这样如果有这种 while(waitpid(-1,&stat,WNOHANG)>0);
这样就能接受到所有的子进程的SIGCHLD了。 wait只执行一次 而Linux信号不排队 同时到达 也只会处罚一次哦 这个必须要了解 这样通过waitpid循环来解决掉,但是如果不指定WNOHANG的话 又比较麻烦 主进程会阻塞 这不是我们所想要的哦 所以必须解阻塞。
引言:
以前对wait waitpid 以及exit这几个函数只是大致上了解,但是看REDIS的AOF和RDB 2种持久化时 均要处理子进程运行完成退出和父进程需要做的什么事情,所以特定看了UNIX环境编程和LINUX系统编程这2本书 重新梳理下整个要点。
内容:
一般而言: 如果程序类似于下面的情况:
if((pid=fork())==0) { dochildtthing(); exit(0); } else if(pid>0) { dofathertthing(); }
子进程就会调用了我们进程退出系统调用exit()分为2种状态 0: 成功 非0:失败
exit需要做以下事情:
1: 关闭所有打开的IO流
2 删除所有的临时文件
1 2是传说中用户态需要完成的事情 接下来就调用_exit() 【这个函数在用户态下 是被exit()封装起来的函数】触发内核来完成进程退出的相关工作
接下来就是清空所有的子进程所需要的资源哦 最后完成了 就会发送一个进程退出状态【信号】给父进程 父进程做什么处理就是编程人员完成的。
子进程发送的信号是SIGCHLD 一般而言就像REDIS fork()了一个进程之后 子进程完成了相应的持久化动作之后,父进程需要了解子进程一些状态来做相应的处理工作,所以在LINUX系统中,这个子进程退出后 还有一个标记位 我们称作zombie进程。父进程必须来取子进程的相应zombie的相应标记信息 不然是不会消除的。
所以对于父进程而言,必须调用相应的函数【首先是注册了SIGCHLD】处理这些标记位哦 wait/waitpid就是干这种事情。我们来看看这系列的调用所要做的事情:
#include <unistd.h> #include <stdio.h> #include <sys/types.h> #include <sys/wait.h> #include <stdlib.h> int main() { int status; pid_t pid; if(fork()==0) { abort(); } pid=wait(&status); printf("pid=%d\n",pid); if(WIFEXITED(status)) printf("正常退出\n"); if(WIFSIGNALED(status)) printf("killed by signal\n"); return 0; }
wait等待子进程挂 结果可以读出子进程是被信号量终止了 所以说 读出子进程的状态还是非常有实际意义的~!
接下来看看wait/waitpid函数的区别:
如果有多个子进程 我该怎么办 要么选择多个wait 但是同时到达 可能就会处理一个子进程 而其他进程状态没有被处理 造成了很多zombie进程 这是不行的 这样就有了waitpid函数:
waitpid(pid_t pid,int *status,int options)
pid=-1 等待任意一子进程
>0 要传入的那个子进程
参数:
WNOHANG: 不会阻塞 立刻返回。 其他参数暂时不去研究了
这样如果有这种 while(waitpid(-1,&stat,WNOHANG)>0);
这样就能接受到所有的子进程的SIGCHLD了。 wait只执行一次 而Linux信号不排队 同时到达 也只会处罚一次哦 这个必须要了解 这样通过waitpid循环来解决掉,但是如果不指定WNOHANG的话 又比较麻烦 主进程会阻塞 这不是我们所想要的哦 所以必须解阻塞。
相关文章推荐
- ACL编程之父子进程机制,父进程守护子进程以防止子进程异常退出
- tomcat java.lang.OutOfMemoryError: Java heap space 问题解决;Java垃圾回收机制详解和调优相关链接
- OpenStack块存储nova-volume工作机制和相关问题
- Redis的持久化机制
- redis中hset相关问题,redis与缓存区别
- 关于使用WindowsUpdate 或 Windows 自动升级时碰到的 svchost.exe 进程 CPU 资源占用过高的问题的相关信息
- redis持久化机制
- 构建高性能.NET应用之配置高可用IIS服务器-第五篇 IIS常见问题之:工作进程回收机制(中)
- 构建高性能.NET应用之配置高可用IIS服务器-第四篇 IIS常见问题之:工作进程回收机制(上)
- redis 2.6 新功能介绍及安装相关问题
- 程序关闭进程没有退出的问题
- ACL编程之父子进程机制,父进程守护子进程以防止子进程异常退出
- 关于判断CreateProcess创建的子进程何时退出和CRichEditCtrl::FindText()一直返回-1的问题解决。
- Redis实战《红丸出品》4.4 Redis高级实用特性之持久化机制
- Redis持久化之大数据服务暂停问题
- ACL编程之父子进程机制,父进程守护子进程以防止子进程异常退出
- 新遇到的问题 , 进程退出代码是 '0xffffffff'
- Linux下调用fork或system启动子进程的信号和资源释放相关问题
- iis7以fastcgi配置php时出现FastCGI进程意外退出问题解决
- 数据库操作中容易出现的问题及相关安全机制