正常退出与异常退出
1.正常退出
- main函数调用return
- 进程调用exit(),标准C库
- 进程调用_exit()或者_Exit(),属于系统调用
- 进程最后一个线程返回
- 最后一个线程调用pthread_exit
2.异常退出
- 调用abort
- 当进程收到某些信号时,如ctrl+c
- 最后一个线程对取消请求做出响应
不管进程如何终止,最后都会执行内核中的同一段代码。这段代码为相应进程关闭所有打开的描述符,释放它所使用的存储器等。
对上述任意一种终止情形,都希望终止进程能够通知其父进程它是如何终止的。对于三个终止函数(exit、_exit 和 _Exit),实现这一点的方法是,将其退出状态作为参数传给函数。在异常终止情况下,内核(不是进程本身)产生一个指示其异常终止原因的终止状态。在任意一种情况下,该终止进程的父进程都能用wait或waitpid函数取得其最终值。
父进程等待子进程退出
为什么要等待子进程退出(创建子进程的目的):
让子进程去干活,干活要知道到底干完了没有,也就是正常退出与异常退出,父进程等待子进程退出并收集子进程的退出状态。
- WIFEXITED(status) 若为正常终止子进程返回的状态,则为真。对于这种情况可执行
- WEXITSTATUS(status),取子进程传送给exit、_exit或_Exit参数的低8位
- WIFSIGNALED(status)若为异常终止子进程返回的状态,则为真(接收到一个不捕捉的信号)。对于这种情况,可执行WTERMSIG(status),取使子进程终止的信号编号。另外,有些实现(非Sigle UNIX Specification)定义宏WCOREDUMP(status),若已产生终止进程的core文件,则它返回真。
- WIFSTOPPED(status)若为当前暂停子进程的返回状态,则为真。对于这种情况,可执行。
- WSTOPIG(status),取使子进程暂停的信号编号。
- WIFCONTINUED(status)若在作业控制暂停后已经继续的子进程返回了状态,则为真。
注:子进程退出状态不被收集,变成僵死进程(僵尸进程)
相关函数
- pid_t wait(int status)
- pid _t waitpid(pid_t pid, int *status, int options)
- int waitid(idtype_t idtype, id_t id, siginfo_t *infop, int options)
函数作用:
- 如果其所有子进程都还在运行,则阻塞
- 如果一个子进程已终止,正等待父进程获取其终止状态,则取得该子进程的终止状态并 立即返回
- 如果它没有任何子进程,则立即出错返回
函数不同点:
wait使调用者阻塞,waitpid有一个选项,可以使调用者不阻塞。
参数含义:
status参数:
是一个整型数指针,非空时,子进程退出状态放在它所指向的地址中;为空时,不关心退出状态。
waitpid函数中的pid参数:
- pid==-1:等待任一子进程。此时waitpid与wait等效。
- pid>0:等待其进程ID与pid相等的子进程。
- pid==0:等待其组ID等于调用进程组ID的任一子进程。
- pid<0:等起其组ID等于pid绝对值的任一子进程。
waitpid函数中的options参数:
- WCONTINUED:若实现支持作业控制,那么由pid指定的任一子进程在暂停后已经继续,但其状态尚未报告,则返回其状态。
- WNOHANG:若由pid指定的子进程并不是立即可用的,则waitpid不阻塞,此时其返回值为0。
- WUNTRACED:若某实现支持作业控制,而由pid指定的任一子进程已处于暂停状态,并且其状态自暂停以后还未报告过,则返回其状态。WIFSTOPPED宏确定返回值是否对应于一个暂停子进程。
孤儿进程:
父进程如果不等待子进程退出,在子进程之前就结束了自己的“生命”,此时子进程叫做孤儿进程。Linux为了比卖你系统存在过多孤儿进程,init进程收留孤儿进程,变成孤儿进程的父进程。