天天看点

linux 僵尸(defunct)进程和孤儿进程

在fork或者exec函数创建一个新的进程,为了收集新进程的退出状态并防止出现僵尸进程(zombie process),父进程应该调用waitpid或者wait等待子进程退出。

在unix/linux 系统中,一个子进程结束了,但是它的父进程没有等待(调用wait / waitpid)它(前提是它的父进程没有退出,这里面操作系统认为它有父进程,操作系统不会清除该进程), 那么它将变成一个僵尸进程,如果父进程也接着退出的话,操作系统会统一将其清除。

之所以被称为僵尸进程,因为它虽然死掉了,但是在进程表中依然存在。子进程退出后分的内存和其他资源都被释放,但它还是在内核进程表中保留一条,内核在父进程回收子进程的退出状态前一直保留它。有一两个僵尸进程不算什么,但是一旦程序频繁的执行fork 或者exec却又不能收集退出状态,那么最终、将会填满进程表这会影响性能,可能导致系统重启。

#include <stdio.h>
#include <stdlib.h>
#include<unistd.h>
#include<sys/types.h>
#include <string.h>
#include <errno.h>

int main(void)
{
	pid_t pid = fork();
	if (pid == -1)
	{
		printf("%s \n", strerror(errno));
		return -1;
	}
	if (pid == 0)
	{
		printf("child exit!");
		exit(0); //子进程直接退出
	}
	else
	{
		printf("parent sleep 100s\n"); //父进程没有调用waitpid去等待子进程退出
		sleep(100);
	}
	return 0;
}
           
linux 僵尸(defunct)进程和孤儿进程

通过ps命令这里使用ps -u hsc (这里hsc是用户名),我们可以看到zombie_test进程有<defunct>defunct的意思就是已死的,无效的,不存在的。

linux 僵尸(defunct)进程和孤儿进程

如果子进程的父进程已经先结束了,此时而它的一个或多个子进程还在运行,这些子进程将成为孤儿进程,它们将由init进程收养。因为每个进程结束的时候,系统都会扫描当前系统中所运行的所有进程, 看有没有哪个进程是刚刚结束的这个进程的子进程,如果是的话,就由Init 来接管他,成为它的父进程……孤儿进程是正常的,不会给系统带来问题。

/*
 * main.c
 *
 *  Created on: 2015-1-29
 *      Author: hsc
 */

#include <stdio.h>
#include <stdlib.h>
#include<unistd.h>
#include<sys/types.h>
#include <sys/wait.h>
#include <string.h>
#include <errno.h>

int main(void)
{
	pid_t pid = fork();
	if (pid == -1)
	{
		printf("%s \n", strerror(errno));
		return -1;
	}
	if (pid == 0)
	{
		while(1)
		{
			sleep(1);
			printf("child ppid=%d \n",getppid()); //打印出其父进程pid
		}

	}
	else
	{
		printf("parent exit\n"); //父进程直接退出
		exit(0);
	}
	return 0;
}

           
linux 僵尸(defunct)进程和孤儿进程

可以看到父进程退出后子进程的父进程的pid为1(init 进程)。

继续阅读