僵尸进程
(1)僵尸进程概述
什么是僵尸进程?
在Linux系统中,任何一个子进程在调用exit()函数结束运行后,内核会释放该进程的所有资源,包括占用的内存和打开的文件等。
同时,也会留下一个叫做僵尸进程(Zombie)的数据结构,Zombie中存储了该进程的进程号、退出码、退出状态、使用的CPU时间等信息。即僵尸进程是早已死亡的子进程,但在进程表中占了一个位置(slot)。
子进程还会向父进程发送SIGCHLD信号,父进程调用wait()或者waitpid()函数可以将僵尸进程释放(为它收尸)。
父进程在没有释放掉僵尸进程就提前结束了,僵尸进程则会由init进程接管。init进程(PID = 1)会作为它的父进程,为它收尸。
但是如果父进程是一个循环,不会结束,却又没有为SIGCHLD信号绑定处理函数,或者没有调用wait()/waitpid()函数为僵尸进程收尸,则该僵尸进程会一直在系统中存在。
(2)僵尸进程的危害
- 如果系统中存在很多僵尸进程,会一直占用系统资源,进程号会被它们一直占用。
- 这时,有限的进程号将会耗尽,使得系统无法fork新的子进程。
僵尸进程代码如下:
//僵尸进程
#include<iostream>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <unistd.h>
using namespace std;
int main()
{
//创建子进程
pid_t pid = fork();
if(pid<0) //fork失败的情况
{
perror("fork error");
return -1;
}
else if(pid>0)//父进程
{
sleep(100);//保证子进程先推迟,父进程还在运行
cout << "father:[pid]=" << getpid() << "[fpid]=" << getppid() << endl;
}
else if(pid==0) //子进程
{
cout << "child:[pid]=" << getpid() << "[fpid]=" << getppid() << endl;
}
return 0;
}
运行后,发现子进程先退出,父进程没有退出或等待执行,子进程内核区的资源无法回收;结果如下
进行了kill操作仍然无法杀死,此时只要杀死它的父进程,他就会被系统回收。