【校招 --阶段一 系统编程】进程等待

41 篇文章 1 订阅
本文详细介绍了进程等待的重要性,防止僵尸进程和内存泄漏,以及如何通过wait和waitpid函数获取子进程的退出状态和资源回收。通过示例代码展示了正常退出和异常退出时的子进程状态检测,包括子进程的退出码和接受的信号。
摘要由CSDN通过智能技术生成

一、进程等待

进程等待必要性

  • 之前说过,子进程退出,父进程如果不管不顾,就可能造成‘僵尸进程’的问题,进而造成内存泄漏。
  • 另外,进程一旦变成僵尸状态,那就刀枪不入,“杀人不眨眼”的kill -9 也无能为力,因为谁也没有办法杀死一个已经死去的进程。
  • 最后,父进程派给子进程的任务完成的如何,我们需要知道。如,子进程运行完成,结果对还是不对,或者是否正常退出。
  • 父进程通过进程等待的方式,回收子进程资源,获取子进程退出信息

进程等待的方法

wait方法

#include<sys/types.h>
#include<sys/wait.h>
pid_t wait(int*status);
返回值:
成功返回被等待进程pid,失败返回-1。
参数:
输出型参数,获取子进程退出状态,不关心则可以设置成为NULL

waitpid方法

pid_ t waitpid(pid_t pid, int *status, int options);
返回值:
当正常返回的时候waitpid返回收集到的子进程的进程ID;
如果设置了选项WNOHANG,而调用中waitpid发现没有已退出的子进程可收集,则返回0;
如果调用中出错,则返回-1,这时errno会被设置成相应的值以指示错误所在;
参数:
pid:
Pid=-1,等待任一个子进程。与wait等效。
Pid>0.等待其进程ID与pid相等的子进程。
status:
WIFEXITED(status): 若为正常终止子进程返回的状态,则为真。(查看进程是否是正常退出)
WEXITSTATUS(status): 若WIFEXITED非零,提取子进程退出码。(查看进程的退出码)
options:
WNOHANG: 若pid指定的子进程没有结束,则waitpid()函数返回0,不予以等待。若正常结束,则返回该子进
程的ID。

如果子进程已经退出,调用wait/waitpid时,wait/waitpid会立即返回,并且释放资源,获得子进程退
出信息。
如果在任意时刻调用wait/waitpid,子进程存在且正常运行,则进程可能阻塞。
如果不存在该子进程,则立即出错返回

在这里插入图片描述
获取子进程status

waitpid第二个参数是用来获取子进程退出的退出状态的

pid_ t waitpid(pid_t pid, int *status, int options);
status:是一个整形指针,其实在传参的时候这参数是一个输出型参数

int st=0;
waitpid(pid, &st,0)//开始等待,子进程退出,操作系统就会从进程的PCDB中读取信息,保存在status变量中的·
返回之后,st中就保存的是进程退出的信息,int 是32bit,是否正常运行,退出码是多少,退出信号是多少

wait和waitpid,都有一个status参数,该参数是一个输出型参数,由操作系统填充。
如果传递NULL,表示不关心子进程的退出状态信息。
否则,操作系统会根据该参数,将子进程的退出信息反馈给父进程。
status不能简单的当作整形来看待,可以当作位图来看待,具体细节如下图(只研究status低16比特
位):

在这里插入图片描述
测试代码:

#include<iostream>
#include<unistd.h>
#include<stdlib.h>
#include<sys/types.h>
#include<sys/wait.h>
using namespace std;
int main(){
pid_t id = fork();
if(id<0){
cout<<"fork error !"<<endl;
}
 if(id==0){
//childer
int count=0;
while(count<10){
cout<<"child is running "<<count<<endl;
count++;
sleep(1);
}
exit(1);
}
cout<<"father wait before!"<<endl;
int ret1=0;
pid_t ret=waitpid(id,&ret1,0);
cout<<"子进程退出码:"<<((ret1>>8)&0xff)<<endl;
cout<<"子进程接受信号:"<<(ret1&0x7f)<<endl;
if(ret>0){

cout<<"wait success"<<endl;
}else{

cout<<"wait failed"<<endl;
}
cout<<"father weait after!"<<endl;

return 0;
}

程序exit正常退出,退出码是1,没有退出信号将输出的是0结果如下:
在这里插入图片描述
当子进程异常退出测试代码:

#include<iostream>
#include<unistd.h>
#include<stdlib.h>
#include<sys/types.h>
#include<sys/wait.h>
using namespace std;
int main(){
pid_t id = fork();
if(id<0){
cout<<"fork error !"<<endl;
}
 if(id==0){
//childer
int count=0;
while(count<10){
cout<<"child is running "<<count<<endl;
count++;
sleep(1);
}
int a=1/0;
}
cout<<"father wait before!"<<endl;
int ret1=0;
pid_t ret=waitpid(id,&ret1,0);
cout<<"子进程退出码:"<<((ret1>>8)&0xff)<<endl;
cout<<"子进程接受信号:"<<(ret1&0x7f)<<endl;
if(ret>0){

cout<<"wait success"<<endl;
}else{

cout<<"wait failed"<<endl;
}
cout<<"father weait after!"<<endl;

return 0;
}

给子进程一个除0错吴

在这里插入图片描述
显示结果是接受一个信号为8的信号子进程终止了。
在这里插入图片描述
参数有两个检测接口:

WIFEXITED(status): 若为正常终止子进程返回的状态,则为真。(查看进程是否是正常退出):!(status&)0x7f)
WEXITSTATUS(status): 若WIFEXITED非零,提取子进程退出码。(查看进程的退出码):(status>>8)&0xff

int main()
{
pid_t pid;
pid = fork();
if(pid < 0){
printf("%s fork error\n",__FUNCTION__);
return 1;
} else if( pid == 0 ){ //child
printf("child is run, pid is : %d\n",getpid());
sleep(5);
exit(257);
} else{
int status = 0;
pid_t ret = waitpid(-1, &status, 0);
printf("this is test for wait\n");
if( WIFEXITED(status) && ret == pid ){//检测子进程是否接受信号
printf("wait child 5s success, child return code is :%d.\n",WEXITSTATUS(status));
}else{
printf("wait child failed, return.\n");
return 1;
}

在这里插入图片描述

评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

自首的小偷

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值