僵尸进程与孤儿进程

僵死进程

如下代码,使子进程先于父进程退出,并且子进程的资源并没有被系统回收,然后父进程仍处于运行状态

#include <iostream>
#include <unistd.h>

using namespace std;

int main(){
	pid_t pid = fork();
	if(pid < 0){
		perror("fork error");
		exit(-1);
	}
	else if(pid == 0){
		cout << "i am child, pid: "<<getpid() << ", ppid: " << getppid()  << endl;				
	}
	else {
		//sleep(1);  // 保证子进程先执行
		cout << "i am father, pid: " << getpid()  << endl;
		while(1){  // 让父进程一直执行,保证子进程在父进程之前退出
			sleep(1);
		}
	}

	return 0;
}

如图子进程id:1562415处于Z状态,该状态表示的就是僵死状态,而父进程pid:1562414 处于S状态,表示父进程还在运行

所以僵死进程的概念:子进程先于父进程退出,退出的子进程就会变为僵死进程,为Z状态。

僵死进程原因

子进程先于父进程退出,子进程会给父进程发送一个SIGCHLD(17)信号,而父进程收到这个信号会忽略处理,进而父进程并不会回收子进程的退出状态,从而导致子进程的资源没有被回收,导致子进程变为僵死进程;子进程在内核中的进程控制块(子进程的资源)没有被操作系统回收掉,所以可以通过ps命令查到子进程是僵死进程。

通过signal()函数,来捕获子进程退出的信号,检测是否会发生SIGCHLD信号:

#include <iostream>
#include <unistd.h>
#include <signal.h>

using namespace std;

void sig_call_back(int signo){
	cout << "catch the signal:" << signo << endl;
}

int main(){
	signal(SIGCHLD, sig_call_back);

	pid_t pid = fork();
	if(pid < 0){
		perror("fork error");
		exit(-1);
	}
	else if(pid == 0){
		cout << "i am child, pid: "<<getpid() << ", ppid: " << getppid()  << endl;		
	}
	else {
		//sleep(1);  // 保证子进程先执行
		cout << "i am father, pid: " << getpid()  << endl;
		while(1){
			sleep(1);
		}
	}
	return 0;
}

运行结果显示,捕获到了SIGCHLD(17)信号。

如何预防僵死进程

僵死进程不能通过kill -9命令杀掉。因为僵死进程本身是一个已经退出的进程,再进行强杀就相当于是“鞭尸”了。

(1)因为子进程退出的时候会发送一个SIGCHLD信号,所以可以通过捕获这个信号然后调用wait()waitpid()函数去等待子进程的退出,这样就可以将退出了的子进程资源释放掉。

#include <iostream>
#include <unistd.h>
#include <signal.h>
#include <sys/wait.h>

using namespace std;


void sig_call_back(int signo){
	cout << "catch the signal:" << signo << endl;
	wait(NULL);  // 调用wait函数等待子进程退出
}

int main(){
	signal(SIGCHLD, sig_call_back);

	pid_t pid = fork();
	if(pid < 0){
		perror("fork error");
		exit(-1);
	}
	else if(pid == 0){
		cout << "i am child, pid: "<<getpid() << ", ppid: " << getppid()  << endl;		
	}
	else {
		//sleep(1);  // 保证子进程先执行
		cout << "i am father, pid: " << getpid()  << endl;
		while(1){
			sleep(1);
		}
	}

	return 0;
}

如图运行结果,再捕获信号回调函数中调用wait后,子进程的资源被释放了,通过ps查看,没有子进程的信息了。

(2)可以通过杀死父进程,让子进程由僵死进程变为孤儿进程,然后孤儿进程再由init进程(1号进程)回收。

僵死进程的危害:

进程的退出状态必须被维持下去,因为他要告诉关心它的进程(父进程),你交给我的任务,我办的怎么样了。可父进程如果一直不读取,那子进程就一直处于Z状态?

答:是的!

维护退出状态本身就是要用数据维护,也属于进程基本信息,所以保存在task_struct(PCB)中,换句话说, Z状态一直不退出, PCB一直都要维护?

答:是的!

那一个父进程创建了很多子进程,就是不回收,是不是就会造成内存资源的浪费?

答:是的!因为数据结构对象本身就要占用内存,想想C中定义一个结构体变量(对象),是要在内存的某个位置进行开辟空间 。

孤儿进程

孤儿进程是指一个父进程退出后,而它的一个或多个子进程还在运行,那么这些子进程将成为孤儿进程。孤儿进程将被 init 进程(进程号为1)所收养,并且由 init 进程对它们完整状态收集工作,孤儿进程一般不会产生任何危害。

#include <iostream>
#include <unistd.h>
#include <signal.h>
#include <sys/wait.h>

using namespace std;

int main(){
	pid_t pid = fork();
	if(pid < 0){
		perror("fork error");
		exit(-1);
	}
	else if(pid == 0){
		cout << "i am child, pid: "<<getpid() << ", ppid: " << getppid()  << endl;		
		sleep(10);  // 子进程等待10s后退出
		cout << "i am child, pid: "<<getpid() << ", ppid: " << getppid()  << endl;		
	}
	else {
		sleep(1);  // 让父进程先退出
		cout << "i am father, pid: " << getpid()  << endl;
		cout << "father exit" <<endl;
		exit(0);
	}
	return 0;
}

父进程先退出,子进程变为孤儿进程,被1号进程收养:

  • 9
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在 Linux 中,僵尸进程孤儿进程都是指与父进程不再有联系的进程,它们通常是由于进程管理不当或程序错误导致的。 **僵尸进程**是已经完成执行任务,但其父进程还没有来得及处理其退出状态的进程。当进程完成执行后,它的退出状态(也称为退出码或终止状态)会被保存在系统中,直到父进程通过 `wait` 或 `waitpid` 等函数来获取该状态。如果父进程没有处理该状态,那么该进程就会成为僵尸进程,占用系统资源。要清理僵尸进程,可以使用 `kill` 命令向其父进程发送 `SIGCHLD` 信号,或者重新编写程序,使其正确处理子进程的退出状态。 **孤儿进程**是指其父进程已经退出或被终止,但其自身仍在运行的进程。孤儿进程会被 `init` 进程(进程号为 `1`)接管,`init` 进程会定期检查系统中是否有孤儿进程,并且将其的父进程设置为 `init` 进程。要避免孤儿进程的产生,可以在父进程退出之前,等待子进程的退出,或者将子进程的父进程设置为 `init` 进程。 可以使用 `ps` 命令来查看系统中的僵尸进程孤儿进程。使用以下命令可以查看所有僵尸进程: ``` ps aux | grep 'Z' ``` 其中,`aux` 选项用于显示所有进程,`grep 'Z'` 用于查找所有状态为 `Z` 的进程,即僵尸进程。 使用以下命令可以查看所有孤儿进程: ``` ps -ejH ``` 其中,`-e` 选项用于显示所有进程,`-j` 选项用于以层次结构的形式显示进程,`-H` 选项用于显示所有孤儿进程

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值