进程退出场景
进程常见退出方法:
1. 正常终止(可以通过 echo $? 查看进程退出码):
?:保存的是最近一个子进程执行完毕时的退出码
2. 异常退出:
为什么要有退出码?
在多进程环境中,父进程创建子进程的目的是什么?是帮助父进程办事,子进程把事情办得怎么样?父进程是需要知道的。
main函数的返回码是可以被父进程获取的,用来判断子进程main的运行结果,main函数返回0,代表成功,非0,表示失败。这个进程是因为什么原因失败的,我们可以返回不同的数字,用不同数字代表不同的原因。
1. _exit函数?是系统调用:终止调用的进程
说明:虽然status是int,但是仅有低8位可以被父进程所用。所以_exit(-1)时,在终端执行$?发现返回值是255。
exit是系统调用。终止进程时候,不会刷新缓冲区,缓冲区的内容会被丢弃掉。
2. exit函数是库函数:让一个正常的程序终止
exit内部最后也会调用_exit,但在调用_exit之前,还做了其他工作:?
3. return退出:是库函数
根据不同的结果现象,返回不同的退出码。
错误码vs退出码
当失败的时候,用来衡量函数、进程出错是的出错详细原因。
代码异常终止,为什么终止了呢?进程出异常,本质是进程收到了对应的信号,自己终止了。
我们可以手动使用kill 命令给这个进程发送11号信号:段错误,使进程异常终止。
所以,一个进程是否出异常,我们只要看有没有收到信号即可。
通过 wait/waitpid 系统调用的方式,让父进程对子进程进行资源回收的等待过程
#include<sys/types.h>
#include<sys/wait.h>
pid_t wait(int*status);
返回值:
成功:返回被等待进程pid。失败(没有子进程):返回-1。
参数:
输出型参数,获取子进程退出状态,不关心则可以设置成为NULL
关于参数status??
core dump 标志我们这里不关心。?
三种退出方式的status:
这里退出状态和终止信号可以使用位运算获得,退出码也可以通过系统定义的宏来获得:
代码示例:
?执行结果:
wait()函数的作用是等待任意一个子进程结束,父进程这里采用阻塞等待,会等子进程执行完变成僵尸状态,wait()回收后,父进程才会继续执行。
pid_ t waitpid(pid_t pid, int *status, int options);
返回值:
参数:
代码示例:
void Worker(int cnt)
{
printf("I am a child ,pid:%d , cnt :%d\n",getpid(),cnt);
}
int main()
{
pid_t id = fork();
if(id == 0)
{
//child;
int cnt = 5;
while(cnt)
{
Worker(cnt);
cnt--;
sleep(2);
}
exit(0);
}
//father
while(1)//非阻塞轮询
{
int status = 0;
pid_t rid = waitpid(id, &status,WNOHANG);
if(rid>0)
{
//wait success,child quit now;
printf("child quit success,exit code:%d,exit signal:%d\n",(status>>8)&0xff,status&0x7f);
break;
}
else if(rid == 0)
{
//wait success,but child not quit.
printf("child is alive,wait agin,father do other thing....\n");
}
else
{
//wait failed,child unknow
printf("wait failed\n");
break;
}
sleep(1);
}
return 0;
}
运行结果:
相关问题:
父进程如何得知子进程的退出信息呢?
PCB中有exit_code和exit_signal字段,子进程退出的时候,要修改状态Z,并将子进程的退出信号和退出码写入PCB中,让父进程回收读取。
本篇结束!?