关于linux c++守护进程调用调用system函数

我们的程序是守护进程,也就是说在最开始会设置一些信号处理,比如说

...
signal(SIGCHLD, SIG_IGN); //忽略子进程结束的信号
...

在这样的场景之下,我想实现这样的一个功能:

通过调用system函数来执行一些系统命令,并根据系统命令的返回值是不是0来判断命令是否执行成功(正常情况下调用system函数,执行成功 return 0; 执行失败 return 正数;)当我写下

int ret = system("ls -a");
std::cout<<"call system ret="<<ret<<std::endl;

时发现始终 ret = -1 查看errno发现errno=10(errno.10 is: No child processes)

进一步分析:

system的实现大致是这样的:

1)主进程fork一个子进程
2)子进程调用exec函数去执行命令
3)父进程调用waitpid来等来子进程结束

也就是说在调用waitpid的时候子进程退出,父进程收不到他的退出信号只发现这个进程不在了,所以有errno=10 ret = -1
具体原因:

当父进程未调用wait系列函数等待子进程结束且未显示忽略SIGCHLD信号,则子进程将成为僵尸进程。如果忽略SIGCHLD信号,子进程不会成为僵尸进程,子进程结束后将会直接退出并释放资源,等父进程调用waitpid时发现pid进程不存在了

解决办法:

signal(SIGCHLD, SIG_DFL);
int ret = system(cmd);
signal(SIGCHLD, SIG_IGN);

在调用system函数的时候按照默认方式处理SIGCHLD信号。

深入探索

system函数的内部实现是要阻塞SIGCHLD信号,原因是怕system的调用者在调用system函数之前有注册过SIGCHLD的信号处理函数,当产生SIGCHLD信号时system的调用者使用一种wait获取到子进程的终止状态,而system函数就无法获取子进程的终止状态(子进程的终止状态只能被获取一次)
注意一点:不管有没有SIGCHLD信号通知,waitpid都可获取子进程的终止状态

忽略SIGCHLD信号调用system的结果是errno=10 ret=-1,忽略SIGCHLD信号然后阻塞SIGCHLD信号并没有影响errno=10 ret=-1的结果,因为阻塞是暂时不接受这个信号,忽略是接受到这个信号后的一种处理方式

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章