Linux系统编程——特殊进程之僵尸进程
僵尸进程(Zombie Process)
进程已执行结束,但进程的占用的资源未被回收。这种进程称为僵尸进程。
在每一个进程退出的时候,内核释放该进程全部的资源、包含打开的文件、占用的内存等。
可是仍然为其保留一定的信息,这些信息主要主要指进程控制块的信息(包含进程号、退出状态、执行时间等)。直到父进程通过 wait() 或 waitpid() 来获取其状态并释放(详细使用方法,请看《等待进程结束》)。 这样就会导致一个问题,假设进程不调用wait()
或 waitpid() 的话, 那么保留的那段信息就不会释放,其进程号就会一直被占用。可是系统所能使用的进程号是有限的,假设大量的产生僵死进程。将由于没有可用的进程号而导致系统不能产生新的进程.此即为僵尸进程的危害,应当避免。子进程已执行结束,父进程未调用 wait() 或 waitpid() 函数回收子进程的资源是子进程变为僵尸进程的原因。
僵尸进程測试程序例如以下:
#include <stdio.h>
#include <unistd.h>
#include <errno.h>
#include <stdlib.h> int main(int argc, char *argv[])
{
pid_t pid;
pid = fork(); //创建进程 if( pid < 0 ){ // 出错
perror("fork error:");
exit(1);
}else if( 0 == pid ){ // 子进程 printf("I am child process.I am exiting.\n");
printf("[son id]: %d\n", getpid() ); exit(0);
}else if( pid > 0){ // 父进程
// 父进程没有调用 wati() 或 watipid()
sleep(1); // 保证子进程先执行
printf("I am father process.I will sleep two seconds\n");
printf("[father id]: %d\n", getpid() ); while(1); // 不让父进程退出
} return 0;
}执行结果:
在终端敲:ps -ef | grep defunct。后面尖括号中是 defunct 的都是僵尸进程。
我们另启一个终端,查看进程的状态。有哪些是僵尸进程:
或者:用ps -e
怎样避免僵尸进程?
1)最简单的方法,父进程通过 wait() 和 waitpid() 等函数等待子进程结束,可是,这会导致父进程挂起。
详细使用方法,请看《进程的控制:结束进程、等待进程结束》。
2)假设父进程要处理的事情非常多。不可以挂起,通过 signal() 函数人为处理信号 SIGCHLD , 仅仅要有子进程退出自己主动调用指定好的回调函数,由于子进程结束后, 父进程会收到该信号 SIGCHLD ,可以在其回调函数里调用 wait() 或 waitpid() 回收。
关于信号的更具体使用方法,请看《信号中断处理》。
測试代码例如以下:#include <stdio.h>
#include <unistd.h>
#include <errno.h>
#include <stdlib.h>
#include <signal.h> void sig_child(int signo)
{
pid_t pid; //处理僵尸进程, -1 代表等待随意一个子进程, WNOHANG代表不堵塞
while( (pid = waitpid(-1, NULL, WNOHANG)) > 0 ){
printf("child %d terminated.\n", pid);
}
} int main()
{
pid_t pid; // 创建捕捉子进程退出信号
// 仅仅要子进程退出,触发SIGCHLD。自己主动调用sig_child()
signal(SIGCHLD, sig_child); pid = fork(); // 创建进程 if (pid < 0){ // 出错
perror("fork error:");
exit(1);
}else if(pid == 0){ // 子进程
printf("I am child process,pid id %d.I am exiting.\n",getpid());
exit(0); }else if(pid > 0){ // 父进程
sleep(2); // 保证子进程先执行
printf("I am father, i am exited\n\n");
system("ps -ef | grep defunct"); // 查看有没有僵尸进程 } return 0;
}执行结果:
3)假设父进程不关心子进程什么时候结束。那么能够用signal(SIGCHLD, SIG_IGN)通知内核,自己对子进程的结束不感兴趣,父进程忽略此信号,那么子进程结束后,内核会回收。 并不再给父进程发送信号。关于信号的更具体使用方法,请看《信号中断处理》。
#include <stdio.h>
#include <unistd.h>
#include <errno.h>
#include <stdlib.h>
#include <signal.h> int main()
{
pid_t pid; // 忽略子进程退出信号的信号
// 那么子进程结束后,内核会回收。 并不再给父进程发送信号
signal(SIGCHLD, SIG_IGN); pid = fork(); // 创建进程 if (pid < 0){ // 出错
perror("fork error:");
exit(1);
}else if(pid == 0){ // 子进程
printf("I am child process,pid id %d.I am exiting.\n",getpid());
exit(0); }else if(pid > 0){ // 父进程
sleep(2); // 保证子进程先执行
printf("I am father, i am exited\n\n");
system("ps -ef | grep defunct"); // 查看有没有僵尸进程 } return 0;
}执行结果:
4)另一些技巧。就是 fork() 两次,父进程 fork() 一个子进程,然后继续工作,子进程 fork() 一 个孙进程后退出,那么孙进程被 init 接管,孙进程结束后,init (1 号进程)会回收。只是子进程的回收还要自己做。《UNIX环境高级编程》8.6节说的很具体。原理是将子进程成为孤儿进程。从而其的父进程变为 init 进程(1 号进程),通过 init 进程(1 号进程)能够处理僵尸进程。很多其它详情。请看《特殊进程之孤儿进程》。
源代码例如以下:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h> int main()
{
pid_t pid;
//创建第一个子进程
pid = fork(); if (pid < 0){ // 出错
perror("fork error:");
exit(1);
}else if (pid == 0){//子进程 //子进程再创建子进程
printf("I am the first child process.pid:%d\tppid:%d\n",getpid(),getppid());
pid = fork();
if (pid < 0){
perror("fork error:");
exit(1);
}else if(pid == 0){ // 子进程
//睡眠3s保证以下的父进程退出,这样当前子进程的父亲就是 init 进程
sleep(3);
printf("I am the second child process.pid: %d\tppid:%d\n",getpid(),getppid());
exit(0); }else if (pid >0){ //父进程退出
printf("first procee is exited.\n");
exit(0);
} }else if(pid > 0){ // 父进程 // 父进程处理第一个子进程退出,回收其资源
if (waitpid(pid, NULL, 0) != pid){
perror("waitepid error:");
exit(1);
} exit(0);
} return 0;
}执行结果:
全部源代码下载:http://download.csdn.net/download/lianghe_work/8999129
Linux系统编程——特殊进程之僵尸进程的更多相关文章
- Linux系统编程(26)——守护进程
		
Linux系统启动时会启动很多系统服务进程,比如inetd,这些系统服务进程没有控制终端,不能直接和用户交互.其它进程都是在用户登录或运行程序时创建,在运行结束或用户注销时终止,但系统服务进程不受用户 ...
 - linux系统编程之框架
		
linux系统编程之框架: 1. 进程 1.1 进程概念 1.1.1 PCB 1.1.2 环境变量 1.2 进程控制 1.3 进程间通信 1.3.1 管道 1.3.2 有名管道 1.3.3 共享内存 ...
 - linux系统编程之进程(三):进程复制fork,孤儿进程,僵尸进程
		
本节目标: 复制进程映像 fork系统调用 孤儿进程.僵尸进程 写时复制 一,进程复制(或产生) 使用fork函数得到的子进程从父进程的继承了整个进程的地址空间,包括:进程上下文.进程堆栈. ...
 - Linux系统编程----僵尸进程
		
什么是僵尸进程? 僵尸进程, 指子进程退出后, 父进程还没有回收子进程的资源,这个子进程就处在于僵尸状态. 来看看如何产生? #include <stdio.h> #include < ...
 - Linux系统编程@进程通信(一)
		
进程间通信概述 需要进程通信的原因: 数据传输 资源共享 通知事件 进程控制 Linux进程间通信(IPC)发展由来 Unix进程间通信 基于System V进程间通信(System V:UNIX系统 ...
 - 020_Linux的孤儿进程与僵尸进程(Unix系统编程)
		
1.前言 之前在看<unix环境高级编程>第八章进程时候,提到孤儿进程和僵尸进程,一直对这两个概念比较模糊.今天被人问到什么是孤儿进程和僵尸进程,会带来什么问题,怎么解决,我只停留在概念上 ...
 - linux系统编程之进程(六):父进程查询子进程的退出,wait,waitpid
		
本节目标: 僵进程 SIGCHLD wait waitpid 一,僵尸进程 当一个子进程先于父进程结束运行时,它与其父进程之间的关联还会保持到父进程也正常地结束运行,或者父进程调用了wait才告终止. ...
 - Linux系统编程@进程管理(一)
		
课程目标: 构建一个基于主机系统的多客户即时通信/聊天室项目 涉及的理论知识 进程控制:僵尸进程/孤儿进程.进程控制.守护进程... 进程间通信:管道.命名管道.信号... 多线程编程: 锁.信号量. ...
 - linux系统编程之进程(一)
		
今天起,开始学习linux系统编程中的另一个新的知识点----进程,在学习进程之前,有很多关于进程的概念需要了解,但是,概念是很枯燥的,也是让人很容易迷糊的,所以,先抛开这些抽象的概念,以实际编码来熟 ...
 
随机推荐
- JAVA中list,set,map与数组之间的转换详解
			
package test; import java.util.*; /** * Created by ming */ public class Test { public static void ma ...
 - [转]asp.net MVC 常见安全问题及解决方案
			
本文转自:http://www.cnblogs.com/Jessy/p/3539564.html asp.net MVC 常见安全问题及解决方案 一.CSRF (Cross-site request ...
 - Android 检查手机上是否安装了指定的软件(根据包名检测)
			
Android检查手机上是否安装了指定的软件(根据包名检测) /** * 检查手机上是否安装了指定的软件 * @param context * @param packageName * @return ...
 - Android  Eclipse 安装教程   2016.06.13版
			
2016.8.16修改 第一步,也是最为关键的一步——修改hosts文件 为什么说是最关键的一步呢?因为接下来的操作,我们都需要连接google网,也就是要连接国外的网站.一般情况下,国外的网站是无法 ...
 - Burn Down Chart(2018.6.4~2018.6.10)
			
Burn Down Chart (2018.6.4~2018.6.10) 娄雨禛[前端部分] 曾子轩[后端部分+燃尽图] 前端 1. 娄雨禛+李鑫 1)在总工程中完成跳转,实现图片显示,并发布到Git ...
 - CSS——半透明
			
1.opacity:不仅背景半透明,内部其他元素也半透明 2.rgba():只有背景半透明. <!DOCTYPE html> <html lang="en"> ...
 - 【译】x86程序员手册21-6.3.5为操作系统保留的指令
			
6.3.5 Some Instructions are Reserved for Operating System 为操作系统保留的一些指令 Instructions that have the po ...
 - Java编译器、JVM、解释器
			
Java虚拟机(JVM)是可运行Java代码的假想计算机.只要根据JVM规格描述将解释器移植到特定的计算机上,就能保证经过编译的任何Java代码能够在该系统上运行.本文首先简要介绍从Java文件的编译 ...
 - 【sqli-labs】 less55 GET -Challenge -Union -14 queries allowed -Variation1 (GET型 挑战 联合查询 只允许14次查询 变化2)
			
http://192.168.136.128/sqli-labs-master/Less-55/?id=1' 试了几次,整型带括号正常了 http://192.168.136.128/sqli-lab ...
 - PHP 之CURL请求封装GET、POST、PUT、DELETE
			
/** * @Description: curl请求 * @Author: Yang * @param $url * @param null $data * @param string $method ...