僵尸进程 wait() waitpid()
来源:互联网 发布:pp语音软件 编辑:程序博客网 时间:2024/05/16 10:01
如果我们已经了解了父进程和子进程的概念,并已经掌握了系统调用exit的用法,但可能很少有人意识到, 在一个进程调用了exit之后,该进程并非马上就消失掉,而是留下一个称为僵尸进程(Zombie)的数据结构。在Linux进程的5种状态中,僵尸进程 是非常特殊的一种,它已经放弃了几乎所有内存空间,没有任何可执行代码,也不能被调度,仅仅在进程列表中保留一个位置,记载该进程的退出状态等信息供其他 进程收集,除此之外,僵尸进程不再占有任何内存空间。从这点来看,僵尸进程虽然有一个很酷的名字,但它的影响力远远抵不上那些真正的僵尸兄弟,真正的僵尸 总能令人感到恐怖,而僵尸进程却除了留下一些供人凭吊的信息,对系统毫无作用。
也许读者们还对这个新概念比较好奇,那就让我们来看一眼Linux里的僵尸进程究竟长什么样子。
当一个进程已退出,但其父进程还没有调用系统调用wait(稍后介绍)对其进行收集之前的这段时间里,它会一直保持僵尸状态,利用这个特点,我们来写一个简单的小程序:
/* zombie.c */sleep的作用是让进程休眠指定的秒数,在这60秒内,子进程已经退出,而父进程正忙着睡觉,不可能对它进行收集,这样,我们就能保持子进程60秒的僵尸状态。
#i nclude
#i nclude
main()
{
pid_t pid;
pid=fork();
if(pid<0) /* 如果出错 */
printf("error occurred!n");
else if(pid==0) /* 如果是子进程 */
exit(0);
else /* 如果是父进程 */
sleep(60); /* 休眠60秒,这段时间里,父进程什么也干不了 */
wait(NULL); /* 收集僵尸进程 */
}
编译这个程序:
$ cc zombie.c -o zombie后台运行程序,以使我们能够执行下一条命令:
$ ./zombie &列一下系统内的进程:
[1] 1577
$ ps -ax没有出现Z的zombie
... ...
1177 pts/0 S 0:00 -bash
1577 pts/0 S 0:00 ./zombie
1578 pts/0 Z 0:00 [zombie ]
1579 pts/0 R 0:00 ps -ax
看到中间的"Z"了吗?那就是僵尸进程的标志,它表示1578号进程现在就是一个僵尸进程。
我们已经学习了系统调用exit,它的作用是使进程退出,但也仅仅限于将一个正常的进程变成一个僵尸进程,并不能将其完全销毁。僵尸进程虽然对其他进程几 乎没有什么影响,不占用CPU时间,消耗的内存也几乎可以忽略不计,但有它在那里呆着,还是让人觉得心里很不舒服。而且Linux系统中进程数目是有限制 的,在一些特殊的情况下,如果存在太多的僵尸进程,也会影响到新进程的产生。那么,我们该如何来消灭这些僵尸进程呢?
先来了解一下僵尸进程的来由,我们知道,Linux和UNIX总有着剪不断理还乱的亲缘关系,僵尸进程的概念也是从UNIX上继承来的,而UNIX的先驱 们设计这个东西并非是因为闲来无聊想烦烦其他的程序员。僵尸进程中保存着很多对程序员和系统管理员非常重要的信息,首先,这个进程是怎么死亡的?是正常退 出呢,还是出现了错误,还是被其它进程强迫退出的?其次,这个进程占用的总系统CPU时间和总用户CPU时间分别是多少?发生页错误的数目和收到信号的数 目。这些信息都被存储在僵尸进程中,试想如果没有僵尸进程,进程一退出,所有与之相关的信息都立刻归于无形,而此时程序员或系统管理员需要用到,就只好干 瞪眼了。
那么,我们如何收集这些信息,并终结这些僵尸进程呢?就要靠我们下面要讲到的waitpid调用和wait调用。这两者的作用都是收集僵尸进程留下的信息,同时使这个进程彻底消失。下面就对这两个调用分别作详细介绍。
对于进程的一生可以用一些形象的比喻作一个小小的总结:
随着一句fork,一个新进程呱呱落地,但它这时只是老进程的一个克隆。
然后随着exec,新进程脱胎换骨,离家独立,开始了为人民服务的职业生涯。
人有生老病死,进程也一样,它可以是自然死亡,即运行到main函数的最后一个”}”,从容地离我们而去;也可以是自杀,自杀有2种方式,一种是调用 exit函数,一种是在main函数内使用return,无论哪一种方式,它都可以留下遗书,放在返回值里保留下来;它还甚至能可被谋杀,被其它进程通过 另外一些方式结束他的生命。
进程死掉以后,会留下一具僵尸,wait和waitpid充当了殓尸工,把僵尸推去火化,使其最终归于无形。
在linux中wait系统调用一文中介绍了其中的一个殓尸工wait, 下面介绍另一个waitpid,这个貌似复杂些。
waitpid函数原型:
#include<sys/wait.h>
pid_t waitpid(pid_t pid,int* status,int options);
也许读者们还对这个新概念比较好奇,那就让我们来看一眼Linux里的僵尸进程究竟长什么样子。
当一个进程已退出,但其父进程还没有调用系统调用wait(稍后介绍)对其进行收集之前的这段时间里,它会一直保持僵尸状态,利用这个特点,我们来写一个简单的小程序:
/* zombie.c */
#i nclude
#i nclude
main()
{
pid_t pid;
pid=fork();
if(pid<0) /* 如果出错 */
printf("error occurred!n");
else if(pid==0) /* 如果是子进程 */
exit(0);
else /* 如果是父进程 */
sleep(60); /* 休眠60秒,这段时间里,父进程什么也干不了 */
wait(NULL); /* 收集僵尸进程 */
}
sleep的作用是让进程休眠指定的秒数,在这60秒内,子进程已经退出,而父进程正忙着睡觉,不可能对它进行收集,这样,我们就能保持子进程60秒的僵尸状态。
编译这个程序:
$ cc zombie.c -o zombie
后台运行程序,以使我们能够执行下一条命令:
$ ./zombie &
[1] 1577
列一下系统内的进程:
$ ps -ax
... ...
1177 pts/0 S 0:00 -bash
1577 pts/0 S 0:00 ./zombie
1578 pts/0 Z 0:00 [zombie ]
1579 pts/0 R 0:00 ps -ax
没有出现Z的zombie
随着一句fork,一个新进程呱呱落地,但它这时只是老进程的一个克隆。
然后随着exec,新进程脱胎换骨,离家独立,开始了为人民服务的职业生涯。
人有生老病死,进程也一样,它可以是自然死亡,即运行到main函数的最后一个”}”,从容地离我们而去;也可以是自杀,自杀有2种方式,一种是调用 exit函数,一种是在main函数内使用return,无论哪一种方式,它都可以留下遗书,放在返回值里保留下来;它还甚至能可被谋杀,被其它进程通过 另外一些方式结束他的生命。
进程死掉以后,会留下一具僵尸,wait和waitpid充当了殓尸工,把僵尸推去火化,使其最终归于无形。
return waitpid(-1,wait_stat,0);
- /* waitpid.c */
- #include<sys/types.h>
- #include<sys/wait.h>
- #include<unistd.h>
- #include<stdio.h>
- int main()
- {
- pid_t pc,pr;
- pc=fork();
- if (pc<0)/* fork错误*/
- {
- printf("fork error\n");
- exit(1);
- }
- else if(pc==0)/*在子进程中*/
- {
- sleep(10);
- exit(0);
- }
- else
- {
- do
- {/* 使用了WNOHANG参数,waitpid不会在这里等待 */
- pr=waitpid(pc,NULL,WNOHANG);
- if (pr==0)
- {
- printf("No child exit\n");
- sleep(1);
- }
- }while (pr==0);
- if (pr==pc)
- printf("successfully get child %d\n",pr);
- else
- printf("wait child error\n");
- }
- return 0;
- }
编译并运行:
$ ./waitpid
No child exit
No child exit
No child exit
No child exit
No child exit
No child exit
No child exit
No child exit
No child exit
No child exit
successfully get child 4607
- 僵尸进程 wait() waitpid()
- 僵尸进程 wait() waitpid()
- 僵尸进程 wait() waitpid()
- 僵尸进程 wait waitpid
- wait waitpid WNOHANG 僵尸进程
- wait、waitpid及僵尸进程
- 僵尸进程 wait() waitpid() 【转 来源不详】
- 僵尸进程以及wait和waitpid函数
- 【读书笔记】僵尸进程和wait,waitpid
- UNIX多进程 - 销毁僵尸进程 - wait()和waitpid()函数
- UNIX多进程 - 销毁僵尸进程 - wait()和waitpid()函数
- 进程编程中的孤儿和僵尸进程--wait/waitpid函数
- (二十五)进程——wait与waitpid、僵尸进程与孤儿进程
- APUE:进程:wait、waitpid
- 进程等待wait,waitpid
- 进程wait()与waitpid()
- linux系统编程之进程(四):wait/waitpid函数与僵尸进程、fork 2 times
- linux系统编程之进程(四):wait/waitpid函数与僵尸进程、fork 2 times
- 僵尸进程 wait() waitpid()
- [iOS] Code Sign error: Provisioning profile XXXX can't be found
- android源码下载
- 远200网敌批评彼事
- 一致性哈希(2)
- 僵尸进程 wait() waitpid()
- 因此那表‘被过分辱溺’的倭乌猩猩们易以少没稠密的毛收
- 一步一步走进Linux HOOK API(六)
- ubuntu使用宽带上网
- VMWare网络的三种工作模式--bridged, host-only, NAT
- 一步一步走进Linux HOOK API(七)
- iPad2越狱告破!!JailbreakMe完美越狱超详细教程
- c#.net 4 json序列化与反序列化通用类
- iPad2越狱