Linux 僵尸进程与wait()函数

wait()函数

当fork启动一个子进程之后,有时,我们想知道子进程何时结束,因此,我们可以通过调用wait()函数让父进程等待子进程的结束。
函数原型为

#include<sys/types.h>
#include<sys/weait.h>

pid_t wait(int* stat_loc);

wait系统调用将暂停父进程直到它的子进程结束为止。此调用返回的时子进程的PID,通常是已经结束的子进程的PID。状态信息允许父进程了解子进程的退出状态,即子进程main函数返回的值或exit函数的退出码。如果stat_loc不是空指针的话,状态信息将被写入它所指向的位置,我们可以用sys/wait.h文件中定义的宏来解释状态信息,如下图示:
在这里插入图片描述

#include<stdio.h>
#include<assert.h>
#include<stdlib.h>
#include<sys/types.h>
#include<unistd.h>
#include<sys/wait.h>
int main()
{
	pid_t pid;
	char* message;
	int n;

	printf("fork program starting!\n");
	pid = fork();
	assert(pid != -1);

	if (pid == 0)
	{
		message = "child";
		n = 3;
	}
	else
	{
		message = "father";
		n = 7;

		int val;
		pid_t _pid = wait(&val);
		if (WIFEXITED(val)) //如果子进程正常结束,它就取一个非零值
		{
			printf("val = %d\n",WEXITSTATUS(val));//如果WIFEXITED非零,他返回子进程的退出码
		}
	}
	
	for (int i = 0; i<n; i++)
	{
		printf("%s\n",message);
		//printf("cur_pid=%d,cur_ppid=%d,message=%s,n=%d\n",getpid(),getppid(),message,n);
		sleep(1);
	}
	exit(3);//退出码设置为3
}

父进程用wait系统调用将自己的执行挂起,直到子进程的状态信息出现为止,这将发生在子进程调用exit的时候。我们将退出码设置为3。父进程然后继续执行,通过测试wait调用的返回值来判断子进程是否终止。如果是,就从状态信息中提取出子进程的退出码。
在这里插入图片描述

僵尸进程

用fork来创建进程确实很实用,但你必须清楚子进程的运行状况,子进程中终止时,与父进程的关联还保持,直到父进程正常终止或父进程调用wait才告结束。因此,进程表中代表子进程的表项不会立即释放,虽然子进程已经结束,但他仍然存在于系统中,因为他的退出码还需要保存起来,以备父进程调用wait使用。而这时他将会成为一个死进程/僵尸进程。
简单地说就是:子进程先于父进程结束,父进程没有调用wait获取子进程的退出码,子进程变成僵死进程

解决方案:

  • 父进程调用wait();
  • 如果父进程结束,那么init会成为子进程的父进程,然后init调用wait获取退出码,子进程便不会变成僵尸进程

代码实现为上面的代码。

还有一个系统调用是用来等待子进程的结束,是waitpid()函数。
函数原型为

#include<sys/types.h>
#include<sys/weait.h>

pid_t waitpid(pid_t pid, int* stat_loc, int option);

pid指定的是需要等待的子进程的pid,如果stat_loc不是空指针的话,状态信息将被写入它所指向的位置,option参数用来改变waitpid的行为,其中最常用的是WNOHANG,它的作用是防止waitpid调用将调用者的执行挂起。
因此,如果想让父进程周期性的检查某个特定的子进程是否已终止,可以使用如下调用方式;

waitpid(child_pid, (int)* 0, WNOHANG);

如果子进程没有结束或意外终止,它就返回0,否则返回child_pid。如果waitpid失败,将返回-1并设置error。

替换进程映像exec系列函数

exec函数可以把当前进程替换为一个新进程,新进程由path或file参数指定。可以使用exec函数将程序的执行从一个程序切换到另一个程序
详细介绍请看这个博客替换进程映像exec系列函数,写的挺不错的。

信号

此处内容日后有机会再补充。。。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值