)
这里写目录标题
1. 前言
控制一个进程包括如何创建它,如何
终止它,并且如何回收它的资源!
为了回收一个进程的资源,创建这个
进程的父进程必须等待这个子进程
死亡后,处理它的代码和数据
2. 进程等待的必要性
既然要学习进程等待,那么要先知道
为什么要有进程等待?进程等待有啥用?
进程等待的必要性:
-
若子进程退出,而父进程对它不管不顾
此时会有僵尸进程问题,有内存泄漏风险 -
当一个进程变成僵尸进程,那它就刀枪
不入了,因为无法杀掉一个死去的进程 -
父进程创建子进程是为了完成某任务,
父进程需要知道它把任务完成得如何
,所以等待子进程死亡是很有必要的! -
父进程需要等待子进程死亡后,
回收它的代码和数据!
3. 进程等待的方法
我们通过系统调用:wait系统函数
来等待子进程死亡
本篇文章着重讲解waitpid函数
int main()
{
pid_t id = fork();
if(id<0)
{
perror("fork");
exit(1);
}
if(id==0)//子进程代码
{
int count = 5;
while(count)
{
printf("[%d]我是子进程,我的pid是: %d\n",count,getpid());
sleep(1);
count--;
}
exit(0);//子进程执行完代码后退出
}
//父进程代码
waitpid(id,NULL,0);
printf("等待子进程成功!\n");
return 0;
}
4. waitpid的参数status
如果父进程想要知道子进程的退出信息
也就是退出码和退出信号,就要用到这个
输出型参数status
status参数可没有你想的这么简单
它的信息并不是按照整个整数来存储的
如果你学过位图的话,那么下面的内容
会很好理解!
我们之研究status的低16比特位
所以如果想要获取status中的这两个
信息,我们需要使用下面的代码来解析:
退出码:
(status >> 8) & 0xFF
退出信号:
status & 0x7F
int main()
{
pid_t id = fork();
if(id<0)
{
perror("fork");
exit(1);
}
if(id==0)//子进程代码
{
int count = 5;
while(count)
{
printf("[%d]我是子进程,我的pid是: %d\n",count,getpid());
sleep(1);
count--;
}
exit(55);//子进程执行完代码后退出
}
//父进程代码
int status = 0;
waitpid(id,&status,0);
printf("等待子进程成功!\n");
printf("进程退出码: %d,进程退出信号: %d\n",(status >> 8) & 0xFF,status & 0x7F);
return 0;
}
5. 对于status中退出信号的验证
int main()
{
pid_t id = fork();
if(id<0)
{
perror("fork");
exit(1);
}
if(id==0)//子进程代码
{
int count = 15;
while(count)
{
printf("[%d]我是子进程,我的pid是: %d\n",count,getpid());
sleep(1);
count--;
}
exit(55);//子进程执行完代码后退出
}
//父进程代码
int status = 0;
waitpid(id,&status,0);
printf("等待子进程成功!\n");
printf("进程退出码: %d,进程退出信号: %d\n",(status >> 8) & 0xFF,status & 0x7F);
return 0;
进程等待status的信息
可以发现,我使用-9号信号
kill掉进程时,进程的退出信号
就是9,然而当进程由于信号异常
终止时,此时进程退出码是无意义的!
6. waitpid的第三个参数option
前面说到,option默认为0代表父进程
阻塞等待子进程死亡,阻塞等待的意思
就是父进程什么都不干,就在waitpid函数
处停下等待子进程!那么假如父进程想要
干一些自己的事情应该怎样做?
waitpid(pid,&status,WNOHANG);
由于父进程执行非阻塞waitpid时
只要子进程不返回父进程就执行下一步代码
所以使用非阻塞等待时往往会循环访问
int main()
{
pid_t id = fork();
if(id<0)
{
perror("fork");
exit(1);
}
if(id==0)//子进程代码
{
int count = 5;
while(count)
{
printf("[%d]我是子进程,我的pid是: %d\n",count,getpid());
sleep(1);
count--;
}
exit(55);//子进程执行完代码后退出
}
//父进程代码
while(1)//循环访问子进程退出情况
{
int wait = waitpid(id,NULL,WNOHANG);
if(wait>0)//子进程退出成功
{
printf("子进程退出成功,子进程pid: %d\n",wait);
break;
}
else if(wait==0)//子进程还没退出,父进程干自己的事情
{
//此处简单模拟父进程干的事情
printf("我是父进程,我现在要干一些别的事情\n");
}
else //等待子进程退出失败
{
perror("waitpid");
exit(1);
}
sleep(1);
}
return 0;
}
7. 总结以及拓展
了解了进程等待话题后,以后创建
子进程去完成任务时就不怕形成
僵尸进程的问题了,进程的真相离我们
越来越近,请同学们耐心学习!
有了这个宏后,以后直接从status获取
进程退出码时,就不用左右移了!