Linux下新进程可以由fork()函数来创建,父进程会根据fork()函数的返回值(-1,0,大于0)来做出判断。通常情况下,父进程在子进程退出后,做一些“善后”工作(即进行一些资源清理的工作),子进程退出时,只会把打开的文件句柄,内存占用,打开的资源进行释放,但是不会清理进程控制块PCB信息,接下来就是父进程的工作了。
那么如果父进程没有完成好“善后”,会出现什么问题呢?
- 僵尸进程
1. 什么是僵尸进程?
一个子进程在其父进程还没有调用wait()或waitpid()的情况下退出,这个子进程就是僵尸进程。任何一个子进程(init除外)在exit()之后,并非马上就消失掉,而是留下一个称为僵尸进程(Zombie)的数据结构,等待父进程处理。
2. 如何预防僵尸进程?
系统中过多的僵尸进程,会造成资源泄漏,所以预防僵尸进程的产生很有必要。
预防方法:
- 让僵尸进程成为孤儿进程,由init进程回收;(手动杀死父进程);
- 调用fork()两次;
- 捕捉SIGCHLD信号,并在信号处理函数中调用wait函数;
- 让僵尸进程的父进程来回收,父进程每隔一段时间来查询子进程是否结束并回收,调用wait()或者waitpid(),通知内核释放僵尸进程;
//这是一个僵尸进程的例子
#include <stdio.h>
#include <stdlib.h>
int main()
{
pid_t id = fork();
if(id < 0)
{
perror("fork");
return 1;
}
else if(id > 0)
{
//parent
printf("parent[%d] is sleeping...\n", getpid());
sleep(30);
}else
{
printf("child[%d] is begin Z...\n", getpid());
sleep(5);
exit(EXIT_SUCCESS);
}
return 0;
}
- 孤儿进程
1. 什么是孤儿进程?
一个父进程退出, 而它的一个或几个子进程仍然还在运行,那么这些子进程就会变成孤儿进程,孤儿进程将被init进程(进程号为1)所收养,并由init进程对它们完成状态收集的工作。
2. 如何预防孤儿进程?
孤儿进程并没有什么实质性的危害,因为当进程变成了孤儿进程之后,会被init进程(进程号为1)所收养,并由init进程对他们完成状态收集的工作。
//这是一个孤儿进程的例子
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
int main()
{
pid_t id = fork();
if(id < 0)
{
perror("fork");
return 1;
}
else if(id == 0)
{
//child
printf("I am child, pid : %d\n", getpid());
sleep(10);
}
else
{
//parent
printf("I am parent, pid: %d\n", getpid());
sleep(3);
exit(0);
}
return 0;
}