孤儿进程:
一个父进程退出,而它的一个或多个子进程还在运行,那么那些子进程将成为孤儿进程。孤儿进程将被init进程(进程号为1)所收养,并由init进程对它们完成状态收集工作。
我们先来模拟实现一个孤儿进程
#include<stdio.h>
#include<unistd.h>
#include<errno.h>
#include<stdlib.h>
int main()
{
pid_t pid=fork();
if(pid < 0)//创建失败
{
perror("fork error\n");
}
if(pid == 0)//子进程
{
printf("I'm the child process\n");
printf("pid:%d ppid:%d\n",getpid(),getppid());
printf("I will sleep five seconds\n");
sleep(5);
printf("pid:%d ppid:%d\n",getpid(),getppid());
printf("child process is exited.\n");
}
else//父进程
{
printf("pid:%d ppid:%d\n",getpid(),getppid());
printf("I am father process\n");
sleep(1);
printf("father process is exites.\n");
}
return 0;
}
我们在Linux下运行结果为:
我们可以看到在子进程睡眠五秒之后(父进程退出),子进程会被进程号为1 的init进程所领养。
僵尸进程
僵尸状态是一种比较特殊的状态,当进程退出时,父进程没有读取到子进程退出的返回代码时就会产生僵尸进程。
僵尸进程会以终止状态保存在进程表中,并且会一直在等待父进程读取退出状态代码,所以,子进程退出,而父进程还在运行,但是父进程没有读取子进程的状态,这时子进程就会进入僵尸状态(Z状态)。但是如果父进程结束了,那么init进程就会领养这个子进程,从而被清除,但是如果父进程是一个循环不会结束,那么,子进程就会一直保持僵尸状态,这样就是为什么会有僵尸进程。
我们可以用top指令来查看有几个僵尸进程
top:动态查看进程的变化
参数:
-d:后面直接加秒数,表示显示整个进程界面更新到的秒数,如top -d 3每三秒更新一次
-b:以批次的方式执行top
-n:与-b搭配,表示需要几次top输出的结果
-p:指定某个特定的PID进行检测
如上图表示有一个僵尸进程
我们来模拟一个僵尸进程的例子
#include<stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main( void )
{
pid_t pid = fork();
if ( pid ==0 )//child
{
sleep(3);
printf("pid=%d\n", getpid());
exit(2);
}
else //parent
{
printf("parent %d\n", getpid()) ;
while ( 1 )
{
printf("parent");
fflush(stdout);
sleep(1) ;
}
}
}
用通俗一点的语言来说就是,子进程先与父进程退出,父进程没有回收子进程的资源,产生了僵尸进程,若子进程先退出,父进程未回收子进程资源,随后也退出,那么会由init进程回收资源。
清除僵尸进程:
将父进程杀掉,父进程死后,僵尸进程变为孤儿进程,会由init进程负责清除僵尸进程,这样它产生的所有僵尸进程都会被清除。
用kill指令干掉某个进程
查看信号:kill -l
图中标出来的是我们常用的几个
- 1:启动被终止的进程,可让该PID重新读取自己的配置文件
- 9:强制中段一个进程,如果该进程运行到一半(如vim)会产生.filename.swap的半产品文件
- 15:正常结束一个进程
- 18:继续运行该进程
- 19:暂停一个进程
我们用ps -l查看进程
指令kill -9 +进程pid:表示强制中断该进程。
僵尸进程的危害
僵尸进程的存在会一直消耗资源,如果很多就会严重影响到服务器的性能