今天在书上看到了 “僵尸进程与孤儿进程”的描述,又查看了一些资料,对这两种进程又多了点认识,这里简要记录下。
这两种进程粗看好像是差不多的,因为都有可能会被init进程回收,但是总的来说还是有点不一样,不管从产生的方式和危害程序都是不一样的。
僵尸进程与孤儿进程的产生
僵尸进程:子进程先结束,然后父进程又不管他,init进程来管理回收。
孤儿进程:父进程先结束,子进程找不到父了,init进程来回收。
僵尸进程回收:把父进程杀死,然后就由init回收了。
所以,僵尸进程与孤儿进程的产生就看是父进程还是子进程先结束。
僵尸进程
子进程结束时,父进程没有对子进程进行等待,不管他的死活。如果程序中父进程能正常结束还好,因为一旦子进程找不到它的父的话,会由init进程接管进行回收处理。
最悲剧的是,一般父进程都是挂一个循环在那里,不会结束的,这个时候系统发现你的父进程还存在的,然后init就不会管你,就产生僵尸进程了,而且如果产生太多会浪费大量的系统资源。
僵尸进程也不能用kill杀死,因为他的进程已经死了。
以下程序会产生僵尸进程,父进程没有等待子进程。通过ps -ef中,能看到
<?php $pid = pcntl_fork(); if ($pid == -1) { die('fork error'); } else if ($pid > 0) { echo "I'm parent" .PHP_EOL; while(true){ sleep(3); } } else { echo "I'm child".PHP_EOL; }
孤儿进程
顾名思义就是进程成孤儿了,因为它的父进程先结束了,它产生的子进程就懵逼了,找不到它的父了,这个时候就只能被init接管回收了。
以下程序父进程先结束,子进程延时一下,然后posix_getppid()会返回1,表示被init接管了
<?php $pid = pcntl_fork(); if ($pid == -1) { die('fork error'); } else if ($pid > 0) { echo "I'm parent" .PHP_EOL; sleep(1); } else { echo "I'm child,ppid:".posix_getppid().PHP_EOL; sleep(3); echo "I'm child,ppid:" .posix_getppid().PHP_EOL; }
僵尸进程与孤儿进程的危害
孤儿进程它会被init最终回收掉,所以危害相对来说要小很多,但是僵尸进程因为会占用大量的进程号和系统资源,如果父进程一直不结束,那么init进程也接管不了,这样就会一直消耗资源源,所以危害相对要高一些。
编写多进程程序时,父进程还是需要使用wait,waitpid等来等待子进程的结束,从而回收资源。