进程的监控(wait...)
-
wait
系统调用wait等来调用进程的任何一个子进程终止,同时将该子进程的信息存储在status中
#include<sys/wait.h> pid_t wait(int *status) //成功:返回终止的子进程ID 失败:返回-1
如果调用之前并没有子进程终止,那么调用将一直阻塞,直到某一个子进程终止
如果调用进程的所有子进程都已经终止了,那么wait将返回-1,并且将errno置为ECHILD
通常使用如下代码来等待所有子进程退出
while((childPid = wait(NULL)) != -1) continue; if(errno != ECHILD) errExit("wait");
-
waitpid
wait系统调用只要当子进程终止时才能够获取到子进程的相关信息,而waitpid则可以获取到更多信息
进程的状态改变
当以下事件发生其一时,我们就说进程的状态发生改变
- 子进程调用_exit()(或exit())终止
- 子进程收到信号,并且因为该信号的默认行为(core | term)而终止
- 子进程因为信号而停止
- 停止的子进程收到SIGCONT信号而恢复运行
#include<sys/wait.h> pip_t waitpid(pid_t pid, int *status, int options) //成功:返回子进程ID或0,失败:返回-1
pid参数解释:
- 如果pid > 0,表示等待进程ID为pid
- 如果pid = 0,则等待与调用进程位于同一个进程组的所有进程
- 如果pid < -1,则会等待与pid绝对值相等的所有子进程
- 如果pid = -1,则会等待所有子进程
部分options如下:
-
WUNTRACED
返回因信号而停止的子进程以及终止子进程信息
-
WCONTINUED
返回因为收到SIGCONT信号而恢复执行的子进程信息
-
WNOHANG
采取轮询方式(polling),即不进行阻塞
如果没有子进程的状态改变,那么立即返回0
从信号处理程序中终止进程 (感觉没啥用)
值得注意的是,但在信号处理程序中使用_exit()时,调用该函数的进程会终止
如果需要通知父进程自己因为某个信号而终止,那么需要在信号处理程序中做一些操作
void handler(int sig){
//做一些工作
signal(sig, SIG_DFL); //将信号处置设为默认,一般是终止
raise(sig); //自举,重新进行信号处理函数,此时就会正常终止
}
-
waitid
waitid可以进行更为精确的操作
#include<sys/wait.h> int waited(idtype_t idtype, id_t id, siginfo_t *info, int option) //错误:返回-1 正常:0 特殊的WNOHANG选项..?
- 如果idtype为P_ALL,则等待任何进程,忽略id值
- 如果idtype为P_PID, 那么等待ID为id为子进程
- 如果idtype为P_PGID,那么等待进程组ID为id的所有子进程
option...,查手册吧
孤儿进程与僵尸进程
-
孤儿进程
当一个进程的父进程已经终止,而其自身还未终止,那么进程之祖---init(pid = 1)就会收养该进程
-
僵尸进程
当一个进程已经终止,而其父进程仍未使用wait/waitpid...等系统调用将,那么该子进程就被称为僵尸进程
当一个子进程被内核转换为僵尸进程时,内核将释放子进程所把持的大部分系统资源,但是在内核表中仍为其维护者一项条目,记录这该子进程的pid,终止状态,资源的使用数据等
大量的僵尸进程将可能填满进程表,阻碍新进程的创建无法使用SIGKILL这种“必杀”信号来杀死僵尸进程,只能等待父进程的终止,当父进程终止后,init进程会接管这些僵尸进程,然后将他们从内核表中清除
注:当子进程终止时,系统会向他们的父进程来发送SIGCHILD信号