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

作者:ComingFlying 时间:2021-12-04 11:53:45 

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

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

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

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

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

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

方法一:


#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;
}

方法二:


#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;
}

方法三:


#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;
}

方法四:


#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

标签:linux,僵尸进程
0
投稿

猜你喜欢

  • Linux操作系统下配置DNS服务器的方法介绍

    2010-01-16 18:42:00
  • 在Linux系统中使用Vim读写远程文件的命令详解

    2023-11-02 15:56:13
  • 中文网站管理员博客:平常心看待导出链接

    2008-12-30 12:48:00
  • Ubuntu无法联网的解决方法

    2021-07-07 05:57:24
  • VMware vCenter6.7添加ESXI主机的教程图解

    2022-02-26 18:40:00
  • IT经理世界:盛大稳定器 IT激励员工创造力

    2009-10-10 17:20:00
  • 设置UCenter Home好友上限与“150法则”

    2009-04-15 07:18:00
  • 妙用Discuz! X1.5“外地IP注册审核”功能

    2010-12-08 16:53:00
  • 详解CentOS 7下安装Tomcat到服务

    2021-04-02 09:58:44
  • 从用户体验和SEO两个角度谈谈网站的描述

    2009-02-16 13:53:00
  • 建站前修改这5处 织梦SEO更优

    2010-09-19 14:32:00
  • 扫尾:删除Linux系统下的历史记录

    2009-09-09 15:22:00
  • 高质量的页面摘要(Snippet)赏析

    2008-04-19 22:05:00
  • 在vmware虚拟机安装dpdk的详细过程

    2022-01-29 19:26:10
  • Rsync 秒杀一切备份工具,你能手动屏蔽某些目录吗?

    2023-10-12 13:09:12
  • 百度对CMS系统网站的改版决不手软

    2009-04-27 10:46:00
  • [经验]少走弯路的十条忠告

    2007-10-10 12:46:00
  • linux环境部署及docker安装redis的方法

    2022-01-13 04:27:31
  • docker容器中文乱码的解决方法

    2022-03-12 09:31:42
  • 网站被百度K了怎么办?

    2007-11-09 19:20:00
  • asp之家 网站运营 m.aspxhome.com