raksmart活动促销

分享

写回答

发帖

Linux查看某个端口的连接数的方法

国外服务器 国外服务器 2375 人阅读 | 0 人回复

发表于 2018-3-27 09:21:02 | 显示全部楼层 |阅读模式

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. #include "../common/common.h"
  2. int main(void)
  3. {
  4.   pid_t pid;

  5.   if ((pid = fork()) < 0) {
  6.     perror("fork error");
  7.     return EXIT_FAILURE;
  8.   } else if (0 == pid) {
  9.     printf("[%ld] child process is running...\n", (long)getpid());
  10.     _exit(0);
  11.   }

  12.   //sleep(15);

  13.   if (waitpid(pid, NULL, 0) < 0) {
  14.     perror("waitpid error");
  15.     return EXIT_FAILURE;
  16.   }

  17.   for (; ;) {
  18.     pause();
  19.   }
  20.   return EXIT_SUCCESS;
  21. }
复制代码

方法二:

  1. #include <sys/wait.h>
  2. #include "../common/common.h"
  3. int main(void)
  4. {
  5.   pid_t pid;

  6.   if ((pid = fork()) < 0) {
  7.     perror("fork error");
  8.     return EXIT_FAILURE;
  9.   } else if (0 == pid) {
  10.     printf("first child is running..\n");
  11.     /**在第一个子进程中再次fork***/
  12.     if ((pid = fork()) < 0) {
  13.       perror("fork error");
  14.       return EXIT_FAILURE;
  15.     } else if (pid > 0) {/**父进程退出**/
  16.       printf("[%ld] first child is exit...\n", (long)getpid());
  17.       _exit(0);
  18.     }

  19.     sleep(2);/**确保父进程先运行**/
  20.     printf("second process pid: %ld, second process's parent pid: %ld\n", (long)getpid(), (long)getppid());
  21.     //sleep(15);
  22.     printf("[%ld] is exit..\n", (long)getpid());
  23.     _exit(0);
  24.   }

  25.   /***获得第一个子进程的退出状态***/
  26.   if (waitpid(pid, NULL, 0) < 0) {
  27.     perror("waitpid error");
  28.     return EXIT_FAILURE;
  29.   }

  30.   for(;;)
  31.     pause();
  32.   return EXIT_SUCCESS;
  33. }
  34. [b]方法三:[/b]

  35. #include <signal.h>
  36. #include "../common/common.h"
  37. int main(void)
  38. {
  39.   /***显示忽略SIGCHLD信号****/
  40.   if (signal(SIGCHLD, SIG_IGN) == SIG_ERR) {
  41.     perror("signal error");
  42.     return EXIT_SUCCESS;
  43.   }

  44.   pid_t pid;
  45.   int i;
  46.   /**产生10个子进程***/
  47.   for (i=0; i<10; ++i) {
  48.     if ((pid = fork()) < 0) {
  49.       perror("fork error");
  50.       return EXIT_FAILURE;
  51.     } else if (0 == pid) {
  52.       _exit(0);
  53.     }
  54.     sleep(2);
  55.     continue;
  56.   }

  57.   for (; ;)
  58.     pause();
  59.   return EXIT_SUCCESS;
  60. }
复制代码

方法四:

  1. #include <signal.h>
  2. #include <sys/wait.h>
  3. #include "../common/common.h"
  4. void sig_chld(int signo);
  5. int main(void)
  6. {
  7.   /**捕获此信号, 此刻系统会立刻检测是否有次信号产生**/
  8.   if (signal(SIGCHLD, sig_chld) == SIG_ERR) {
  9.     handler_err("signal error to SIGCHLD");
  10.   }

  11.   pid_t pid;
  12.   int i;
  13.   for (i=0; i<10; i++) {

  14.     if ((pid = fork()) < 0) {
  15.       handler_err("fork error");
  16.     } else if (0 == pid) {
  17.       printf("child pid: %d\n", getpid());
  18.       _exit(0);
  19.     }

  20.     sleep(1);
  21.     continue;
  22.   }

  23.   for (; ;) {
  24.     pause();
  25.   }  
  26.   return EXIT_SUCCESS;
  27. }

  28. /**捕获到信号后会立刻执行此段代码***/
  29. void sig_chld(int signo)
  30. {
  31.   printf("receive child signal\n");
  32.   if (waitpid(-1, NULL, 0) < 0) {
  33.     perror("waitpid error");
  34.   }

  35.   if (signal(SIGCHLD, sig_chld) == SIG_ERR) {
  36.     perror("signal error to SIGCHLD");
  37.   }
  38. }
复制代码
您需要登录后才可以回帖 登录 | 注册

本版积分规则