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

最新文章

  1. 利用jQuery对插件进行扩展时,方法$.extend()、$.fn.extend()区别与联系
  2. AES,RSA对称加密和非对称加密
  3. 如何设置unobtrusive的语言包
  4. .NET EXCEL NPOI 图片
  5. MYSQL 日期函数【转】
  6. powerdesigner jdbc 连接 oracle
  7. 【C++之STL】理解容器(ing)
  8. mysql数据库容量查询
  9. PopUpManager弹出窗口
  10. Maven 私服的简单使用
  11. MongoDB 基础(2019年开篇)
  12. Pointer-network的tensorflow实现-1
  13. 2019.02.26 bzoj4311: 向量(线段树分治+凸包)
  14. 微信小程序navigator
  15. 省选模拟赛第四轮 B——O(n^4)-&gt;O(n^3)-&gt;O(n^2)
  16. C++ 智能指针五
  17. Java反射 - 简单的给Bean赋值和取值
  18. flex作图
  19. LeetCode-330.Patching Array
  20. struts2 action重定向action中文乱码处理

热门文章

  1. 设计模式UML模型图
  2. js千分位处理
  3. Python网络爬虫(四)
  4. [Leetcode]008.String to Integer (atoi)
  5. Codeforces Round #565 (Div. 3) A. Divide it!
  6. BZOJ4552(二分+线段树)
  7. IDEA 工具使用指南
  8. Nodejs 连接 mysql时报错 Error: Cannot enqueue Query after fatal error
  9. UnityError 打包到Android错误解决
  10. Hive项目实战:用Hive分析“余额宝”躺着赚大钱背后的逻辑