服务器资讯

防御吧作为15年知名老牌域名服务商,CNNIC和CANN双认证域名注册商,已经
持续为500多万个域名提供服务,包括智能DNS/自由转移/隐私保护等服务!
linux下避免僵尸进程的几种方法
2023-01-30 13:47:49 【

Ansible 点对点命令快速入门指南示例,Linux上定位后台服务偶发崩溃的解决方法,Linux系统已经融入了我们的生活!论Linux系统的重要性,在linux服务器下使用版本控制软件SVN的方法,关于Linux内核编译失败的一些解决方案,linux top命令查看用户内存大小方法,在Linux上使用Wine安装轻聊版的QQ的步骤讲解,linux下磁盘查看命令分享Linux,僵尸进程,详解linux下避免僵尸进程的几种方法



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");


 }


}



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


】【打印关闭】 【返回顶部
分享到QQ空间
分享到: 
上一篇没有了 下一篇Linux关机时执行指定脚本功能实现

立足首都,辐射全球,免备案服务器网专注云防御及云计算服务15年!

联系我们

服务热线:010-56157787 ,010-56159998
企业QQ:4000043998
技术支持:010-56159998
Copyright ? 2003-2016 gnmianbeian.com. 免备案服务器网版权所有 增值许可:京B2-20140042号
售前咨询
公司总机:4000043998 01056155355
24小时电话:010-56159998
投诉电话:18910191973
值班售后/技术支持
售后服务/财务
备案专员
紧急电话:18610088800