linux进程退出

正常退出:

  • main函数调用return
  • 进程调用exit,标准C库
  • 进程调用_exit()或者_Exit(),属于系统调用
  • 进程最后一个线程返回
  • 最后一个线程调用pthread_exit

异常退出:

  • 调用abort
  • 当进程收到某些信号时,比如ctrl+c
  • 最后一个线程对取消(cancellation)请求做出响应

  不管系统如何终止,最后都会执行内核中的同一段代码。这段代码为相应进程关闭所有打开描述符,释放它所使用的存储器等。

对上述的任意一种终止形式,我们都希望终止进程能够通知其父进程它是如何终止的。对于三个终止函数(exit 、_exit和_Exit),实现这一点的方法是将其退出状态作为参数传给函数。在异常终止情况下,内核(不是进程本身)产生一个指示其异常终止原因的终止状态。在任意一种情况下,该终止进程的父进程都能用wait或waitpid函数取得其终止状态。

父进程等待子进程退出

为什么要等待子进程退出

为了使父进程知道子进程到底有没有正常执行完毕并退出,知晓子进程的执行状态,比如有没有被杀死或发生异常导致退出。

父进程等待子进程退出并收集子进程的退出状态

子进程退出状态不被收集,会变成僵死进程(僵尸进程)

 示例:

 以上代码中子进程只调用了exit进行退出,但是父进程没有对子进程退出的状态进行接收,所以子进程退出之后会变成僵尸进程。可以看到父进程的状态是S+代表正在运行,子进程的状态是Z+代表是僵尸(Zombie)进程。

wait

 status参数:是一个整型数指针,如果该参数是非空,子进程退出状态就存放在它所指向的地址中;如果是空,则父进程不关心退出状态。

  • 如果其所有子进程都还在运行,则阻塞。
  • 如果一个子进程已经终止,正等待父进程获取其终止状态,则取得该子进程的终止状态立即返回
  • 如果它没有任何子进程,则立即返回出错

以上代码加上了wait接收子进程的退出状态,在使用ps指令查看系统进程,可以发现没有僵尸进程了。ps:这里的status直接是写的NULL,所以父进程并没有去关心子进程的退出状态。

解析子进程退出状态的宏

 

使用WEXITSTATUS可以将子进程退出状态解析出来,如果不适用这个宏,直接打印status的结果是不正确的。

waitpid

waitpid与wait的区别:wait使调用者阻塞,waitpid有一个选项,可以使调用者不阻塞。

pid参数作用:当pid为-1时,等待任一子进程。就这一方面,与wait等效;当pid>0,等待其进程ID与pid相等的子进程;当pid为0时,等待其组ID等于调用进程组的任一子进程;当pid<-1时,等待其组ID等于pid绝对值的任一子进程。

options参数:WCONTINUED:若实现支持作业控制,那么由pid指定的任一子进程在暂停后已经继续,但其状态未报告,则返回其状态;WNOHANG:若由pid指定的子进程并不是立即可用的,则waitpid不阻塞,此时其返回值为0;WUNTRACED:若某实现支持作业控制,而由pid指定的任一子进程已经处于暂停状态,并且其状态自暂停以来为报告过,则返回其状态。WIFSTOPPED宏返确定返回值是否对应与一个暂停子进程。

 可以看到使用了waitpid之后父进程不会被阻塞,可以和子进程一起跑

但是我们需要注意一下,使用waitpid之后的子进程会变成僵尸进程,所以一般用的不是很多。如下图所示

孤儿进程:

父进程如果不等待子进程退出,在子进程之前就结束了自己的“声明”,此时子进程就叫做孤儿进程。Linux为了避免系统存在过多的孤儿进程,init进程收留孤儿进程,变成孤儿进程的父进程。

示例:

 

可以看到父进程在子进程退出之前就死掉了,然后子进程在运行时,被init进程接收,成为孤儿进程继续运行。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值