一、僵尸进程
1. 概念
处于僵尸状态的进程就是一个僵尸进程
2. 产生原因
- 子进程先于父进程退出;
- 子进程退出为了保存自己的退出状态,因此退出后资源不会被完全释放。正常情况下,当操作系统检测到了子进程要退出时,会通过 SIGCHLD 信号通知父进程,让父进程去获取子进程的退出状态,此时子进程等待父进程的获取,当父进程获取后,系统才会完全释放子进程的资源;
- 假如父进程没有关心子进程的退出,那么这时候这个退出子进程将处于一个僵尸状态,成为一个僵尸进程,占据着系统资源而不释放。
3. 实例
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
int main()
{
pid_t pid = fork();
if(pid < 0)
{
perror("fork error");
exit(0);
}
else if(pid == 0) // 子进程
{
printf("This is a child process! [PID:%d]\n", getpid());
sleep(5);
exit(0);
}
else // 父进程
{
printf("This is a parent process! [PID:%d]\n", getpid());
sleep(20);
}
return 0;
}
编译运行:
过5秒后,子进程先于父进程退出,这时查看进程信息:
4. 危害
如果子进程变成僵尸进程,那么它的退出状态需要一直维持下去,而维护退出状态也是一种数据维护,需要占用一定的内存空间。倘若一个父进程创建了很多子进程,都不进行资源回收,那么就会导致资源泄露,正常进程可能无法创建,严重的话,还会影响服务器的性能。
5. 处理方法
(1)进程等待(避免方法)
子进程退出时会向父进程发送 SIGCHILD 信号,在父进程处理SIGCHILD信号时,调用进程等待函数 wait() / waitpid(),来获取子进程的退出状态,以避免子进程处于僵尸状态。
(2)僵尸进程->孤儿进程(解决方法)
我们知道僵尸进程用 kill 命令是无法杀掉的,但是我们可以杀掉僵尸进程的Daddy(父进程),这样僵尸进程就成了孤儿进程,而孤儿进程不会占用系统资源,它会被PID为1的 init 进程收养,然后回收其资源。
二、孤儿进程
1. 产生原因
父进程先于子进程退出,那么子进程将成为孤儿进程,并进入后台运行。
2. 特性
孤儿进程并不会一直存在,它会被 init 进程(PID为1)所收养。也就是说,这个孤儿进程如果后来退出了,那么init进程将负责释放其资源,因此,孤儿进程并没有什么危害。
3. 实例
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
int main()
{
pid_t pid = fork();
if(pid < 0)
{
perror("fork error");
exit(0);
}
else if(pid == 0) // 子进程
{
printf("This is a child process! [PID:%d]\n", getpid());
sleep(20);
}
else // 父进程
{
printf("This is a parent process! [PID:%d]\n", getpid());
sleep(5);
exit(0);
}
return 0;
}
编译运行:
过5秒后,父进程先于子进程退出,这时查看进程信息:
并且,此时我们也可以发现子进程的父进程已经变了,它的父进程PID变为1,如下: