@time 2019-07-12
@author Ruo_Xiao
一、产生原因
1、僵尸进程
子进程退出,父进程运行,父进程没有调用 wait 或者 waitpid 函数,那么子进程就处于僵尸状态(Z)。
2、孤儿进程
子进程运行,父进程退出,孤儿进程由 init 进程收养,此时子进程就变成了孤儿进程。
3、系统为什么需要僵尸进程这种进程状态
由于父进程创建子进程是异步的,双方不知道各自的运行状态,而父进程有的时候需要知道子进程退出时的一些信息,所以 linux 提供了一种机制,通过让子进程退出时向父进程发送 SIGCHRD 信号来告知父进程,子进程已经退出了。同时,父进程通过调用 wait 和 waitpid 来获取子进程的退出信息。
二、有什么危害
1、僵尸进程
有很大危害。因为僵尸进程已经挺尸了,对系统没有什么作用,但是依然在进程表占了位置,如果 os 有大量的僵尸进程,那么进程号就会被大量无故占用,严重的话再次 fork 进程可能失败。
2、孤儿进程
没什么危害。因为该进程只是父进程换成了 init ,依然可以正常运行。
三、如何预防(解决方案)
1、kill 父进程
kill 父进程之后,僵尸进程会变成孤儿僵尸进程,由 init 收养,通过 init 是循环 wait ,从而让子进程彻底退出。
2、注册 SIGCHRD 信号的信号处理函数,在函数中调用 wait(调用者堵塞) 或者 waitpid(可以配置调用者不堵塞) 。
3、fork 两次,创建子进程,子进程在创建孙进程,最后 kill 子进程,那么孙进程就由 init 收养。
栗子:
#include <iostream>
#include <signal.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/wait.h>
#include <sys/types.h>
void signalproc(int isig)
{
int istatus;
waitpid(SIG_BLOCK, &istatus, WNOHANG);
}
int main()
{
signal(SIGCHLD, signalproc);
pid_t pid;
int r = fork();
if (r < 0)
std::cout << "创建进程失败!" << std::endl;
else if (r == 0)
{
std::cout << "子进程开始运行!" << std::endl;
int r1 = fork();
if (r < 0)
std::cout << "创建孙进程失败!" << std::endl;
else if (r1 == 0)
{
std::cout << "孙进程开始运行,并进入循环!" << std::endl;
while (true)
{
std::cout << "Sun" << std::endl;
sleep(1);
}
}
std::cout << "子进程退出" << std::endl;
exit(0);
}
std::cout << "父进程开始进入循环!" << std::endl;
while (true)
{
std::cout << "parent" << std::endl;
sleep(1);
}
return 0;
}
结果:
父进程开始进入循环!
parent
子进程开始运行!
子进程退出
孙进程开始运行,并进入循环!
Sun
parent
Sun
parent
Sun
parent
Sun
parent
输入指令:
ps -eo pid,ppid,sid,comm,stat | grep -E 'bash|fork'
结果:
13997 2539 13997 bash Ss
15816 2539 15816 bash Ss
15939 15816 15816 forktest S+
15941 1210 15816 forktest S+
注意:
由于我的系统是 ubuntu 18.04,所以 forktest 的孙进程的父进程是 1210,不是 init 。1210 进程是 systemd 进程,该进程的具体含义后续有时间会说明
(SAW:Game Over!)