详解linux下避免僵尸进程的几种方法

吾爱主题 阅读:206 2024-04-05 13:53:05 评论:0

linux下我们可以调用fork函数创建子进程,创建的子进程将会得到父进程的数据空间、堆、栈......副本(采用写时复制机制),子进程将会继承父进程的信号掩码、信号处理方式、当前工作目录、会话id、组id......。当子进程退出时父进程应当及时获取子进程退出状态,否则,如果父进程是一直在运行,那么子进程的退出状态将一直保存在内存中,直到父进程退出才释放。

我们可以使用如下几种方法避免僵尸进程的产生:

1.在fork后调用wait/waitpid函数取得子进程退出状态。

2.调用fork两次(第一次调用产生一个子进程,第二次调用fork是在第一个子进程中调用,同时将父进程退出(第一个子进程退出),此时的第二个子进程的父进程id为init进程id(注意:新版本Ubuntu并不是init的进程id))。

3.在程序中显示忽略SIGCHLD信号(子进程退出时会产生一个SIGCHLD信号,我们显示忽略此信号即可)。

4.捕获SIGCHLD信号并在捕获程序中调用wait/waitpid函数。

方法一:

?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 #include "../common/common.h" int main( void ) {    pid_t pid;      if ((pid = fork()) < 0) {      perror ( "fork error" );      return EXIT_FAILURE;    } else if (0 == pid) {      printf ( "[%ld] child process is running...\n" , ( long )getpid());      _exit(0);    }      //sleep(15);      if (waitpid(pid, NULL, 0) < 0) {      perror ( "waitpid error" );      return EXIT_FAILURE;    }      for (; ;) {      pause();    }    return EXIT_SUCCESS; }

方法二:

?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 #include <sys/wait.h> #include "../common/common.h" int main( void ) {    pid_t pid;      if ((pid = fork()) < 0) {      perror ( "fork error" );      return EXIT_FAILURE;    } else if (0 == pid) {      printf ( "first child is running..\n" );      /**在第一个子进程中再次fork***/      if ((pid = fork()) < 0) {        perror ( "fork error" );        return EXIT_FAILURE;      } else if (pid > 0) { /**父进程退出**/        printf ( "[%ld] first child is exit...\n" , ( long )getpid());        _exit(0);      }        sleep(2); /**确保父进程先运行**/      printf ( "second process pid: %ld, second process's parent pid: %ld\n" , ( long )getpid(), ( long )getppid());      //sleep(15);      printf ( "[%ld] is exit..\n" , ( long )getpid());      _exit(0);    }      /***获得第一个子进程的退出状态***/    if (waitpid(pid, NULL, 0) < 0) {      perror ( "waitpid error" );      return EXIT_FAILURE;    }      for (;;)      pause();    return EXIT_SUCCESS; }

方法三:

?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 #include <signal.h> #include "../common/common.h" int main( void ) {    /***显示忽略SIGCHLD信号****/    if ( signal (SIGCHLD, SIG_IGN) == SIG_ERR) {      perror ( "signal error" );      return EXIT_SUCCESS;    }      pid_t pid;    int i;    /**产生10个子进程***/    for (i=0; i<10; ++i) {      if ((pid = fork()) < 0) {        perror ( "fork error" );        return EXIT_FAILURE;      } else if (0 == pid) {        _exit(0);      }      sleep(2);      continue ;    }      for (; ;)      pause();    return EXIT_SUCCESS; }

方法四:

?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 #include <signal.h> #include <sys/wait.h> #include "../common/common.h" void sig_chld( int signo); int main( void ) {    /**捕获此信号, 此刻系统会立刻检测是否有次信号产生**/    if ( signal (SIGCHLD, sig_chld) == SIG_ERR) {      handler_err( "signal error to SIGCHLD" );    }      pid_t pid;    int i;    for (i=0; i<10; i++) {        if ((pid = fork()) < 0) {        handler_err( "fork error" );      } else if (0 == pid) {        printf ( "child pid: %d\n" , getpid());        _exit(0);      }        sleep(1);      continue ;    }      for (; ;) {      pause();       return EXIT_SUCCESS; }   /**捕获到信号后会立刻执行此段代码***/ void sig_chld( int signo) {    printf ( "receive child signal\n" );    if (waitpid(-1, NULL, 0) < 0) {      perror ( "waitpid error" );    }      if ( signal (SIGCHLD, sig_chld) == SIG_ERR) {      perror ( "signal error to SIGCHLD" );    } }

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。

原文链接:http://blog.csdn.net/ComingFlying/article/details/76718809

可以去百度分享获取分享代码输入这里。
声明

1.本站遵循行业规范,任何转载的稿件都会明确标注作者和来源;2.本站的原创文章,请转载时务必注明文章作者和来源,不尊重原创的行为我们将追究责任;3.作者投稿可能会经我们编辑修改或补充。

【腾讯云】云服务器产品特惠热卖中
搜索
标签列表
    关注我们

    了解等多精彩内容