wait系统调用

进程的监控(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如下:

    1. WUNTRACED

      返回因信号而停止的子进程以及终止子进程信息

    2. WCONTINUED

      返回因为收到SIGCONT信号而恢复执行的子进程信息

    3. 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信号

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容