孤儿进程的概念:
若子进程的父进程已经死掉,而子进程还存活着,这个进程就成了孤儿进程。
为了保证每个进程都有一个父进程,孤儿进程会被init进程领养,init进程成为了孤儿进程的养父进程,当孤儿进程退出之后,由init进程完成对孤儿进程的回收。
模拟孤儿进程的案例
编写模拟孤儿进程的代码讲解孤儿进程,验证孤儿进程的父进程是否由原来的父进程变成了init进程。
孤儿进程: 父进程先退出, 子进程就变成了孤儿进程, 此时被init进程领养,
当孤儿进程退出之后, 就会被init进程回收.
//模拟孤儿进程 函数测试
#include
#include
#include
#include
#include
int main()
{
//创建子进程
//pid_t fork(void);
pid_t pid = fork();
if(pid<0)
{
perror("fork error");
return -1;
}
else if(pid>0)//父进程
{
/*
man getpid for help
*/
sleep(5);
printf("father: pid==[%d] fpid==[%d]\n",getpid(),getppid());
}
else if(pid == 0)//子进程
{
printf("child: pid==[%d] fpid==[%d]\n",getpid(),getppid());
sleep(10);
printf("child: pid==[%d] fpid==[%d]\n",getpid(),getppid());
}
return 0;
}
$ gcc -g -Wall orphan.c -o orphan
$ ./orphan
child: pid==[95035] fpid==[95034]
father: pid==[95034] fpid==[91160]
child: pid==[95035] fpid==[1] //被init进程回收
僵尸进程的概念:
若子进程死了,父进程还活着, 但是父进程没有调用wait或waitpid函数完成对子进程的回收,则该子进程就成了僵尸进程。
如何解决僵尸进程
由于僵尸进程是一个已经死亡的进程,所以不能使用kill命令将其杀死
通过杀死其父进程的方法可以消除僵尸进程。
杀死其父进程后,这个僵尸进程会被init进程领养,由init进程完成对僵尸进程的回收。
模拟僵尸进程的案例
编写模拟僵尸进程的代码讲解僵尸进程, 验证若子进程先于父进程退出, 而父进程没有调用wait或者waitpid函数进行回收, 从而使子进程成为了僵尸进程.
//模拟僵尸进程 函数测试
#include
#include
#include
#include
#include
int main()
{
//创建子进程
//pid_t fork(void);
pid_t pid = fork();
if(pid<0)
{
perror("fork error");
return -1;
}
else if(pid>0)//父进程
{
/*
man getpid for help
*/
sleep(20);//sleep 让子进程先结束
printf("father: pid==[%d] fpid==[%d]\n",getpid(),getppid());
}
else if(pid == 0)//子进程
{
printf("child: pid==[%d] fpid==[%d]\n",getpid(),getppid());
}
return 0;
}
$ ps -ef | grep zombie
wym 95932 95707 0 23:29 pts/6 00:00:00 grep --color=auto zombie
$ ps -ef | grep zombie
wym 96155 91160 0 23:29 pts/5 00:00:00 ./zombie
wym 96156 96155 0 23:29 pts/5 00:00:00 [zombie] <defunct>
#defunct 表明是僵尸进程,无法接受信号,kill -9 也无法杀死
#如何解决僵尸进程:
应该使用杀死僵尸进程父进程的方法来解决僵尸进程
解决僵尸进程的更好方法是使用进程回收函数
进程回收函数
wait函数
函数原型:
pid_t wait(int *status);
函数作用:
阻塞并等待子进程退出
回收子进程残留资源
获取子进程结束状态(退出原因)。
返回值:
成功:清理掉的子进程ID;
失败:-1 (没有子进程)
status参数:子进程的退出状态 – 传出参数
WIFEXITED(status):为非0 → 进程正常结束
WEXITSTATUS(status):获取进程退出状态
WIFSIGNALED(status):为非0 → 进程异常终止
WTERMSIG(status):取得进程终止的信号编号。
wait函数练习
使用wait函数完成父进程对子进程的回收
//父进程调用wait完成对子进程地回收 函数测试
#include
#include
#include
#include
#include /*for wait*/
#include
int main()
{
//创建子进程
//pid_t fork(void);
pid_t pid = fork();
if(pid<0)
{
perror("fork error");
return -1;
}
else if(pid>0)//父进程
{
printf("father: pid==[%d] fpid==[%d]\n",getpid(),getppid());
int status = 0;
pid_t wpid = wait(&status);
printf("wpid==[%d]\n",wpid);
if(WIFEXITED(status))//正常退出
{
printf("child normal exit,status==[%d]\n",WEXITSTATUS(status));
}else if(WIFSIGNALED(status))//被信号杀死
{
printf("child killed by signal,status==[%d]\n",WTERMSIG(status));
}
}
else if(pid == 0)//子进程
{
printf("child: pid==[%d] fpid==[%d]\n",getpid(),getppid());
sleep(20);
/*
sleep 20 可以测试被信号 15(Terminated) 杀死
$ ps -ef | grep wait
wym 111409 91160 0 00:04 pts/5 00:00:00 ./wait
wym 111410 111409 0 00:04 pts/5 00:00:00 ./wait
wym 111436 95707 0 00:04 pts/6 00:00:00 grep --color=auto wait
$ kill -15 111409
$ ./wait
father: pid==[111409] fpid==[91160]
child: pid==[111410] fpid==[111409]
Terminated
*/
return 9;
/*
* 不使用kill 信号
$ ./wait
father: pid==[112977] fpid==[91160]
child: pid==[112978] fpid==[112977]
wpid==[112978]
child normal exit,status==[9]
*
*/
}
return 0;
}
waitpid函数
函数原型:
pid_t waitpid(pid_t pid, int *status, in options);
函数作用
同wait函数
函数参数
参数:
pid:
pid = -1 等待任一子进程。
pid > 0 等待其进程ID与pid相等的子进程。与wait等效。
pid = 0 等待进程组ID与目前进程相同的任何子进程,也就是说任何和调用
waitpid()函数的进程在同一个进程组的进程。
pid < -1 等待其组ID等于pid的绝对值的任一子进程。(适用于子进程在其他组的情况)
status: 子进程的退出状态,用法同wait函数。
options:设置为WNOHANG,函数非阻塞,设置为0,函数阻塞。
函数返回值
0:返回回收掉的子进程ID;
-1:无子进程
=0:参3为WNOHANG,且子进程正在运行。
waitpid函数练习
使用waitpid函数完成对子进程的回收
//父进程调用waitpid完成对子进程地回收 函数测试
#include
#include
#include
#include
#include /*for wait*/
#include
int main()
{
//创建子进程
//pid_t fork(void);
pid_t pid = fork();
if(pid<0)
{
perror("fork error");
return -1;
}
else if(pid>0)//父进程
{
printf("father: pid==[%d] fpid==[%d]\n",getpid(),getppid());
int status = 0;
//pid_t wpid = waitpid(pid,&status,0);//same to pid_t wpid = wait(&status);
while(1)//循环等待子进程退出
{
pid_t wpid = waitpid(-1,&status,WNOHANG);//等待任一子进程,WNOHANG不阻塞
printf("wpid==[%d]\n",wpid);
if(wpid>0)
{
if(WIFEXITED(status))//正常退出
{
printf("child normal exit,status==[%d]\n",WEXITSTATUS(status));
}else if(WIFSIGNALED(status))//被信号杀死
{
printf("child killed by signal,status==[%d]\n",WTERMSIG(status));
}
}
else if(wpid==0)//子进程还活着
{
printf("child is living,wpid==[%d]\n",wpid);
}
else if(wpid==-1)//没有子进程
{
printf("no child is living,wpid==[%d]\n",wpid);
break;
}
}
}
else if(pid == 0)//子进程
{
printf("child: pid==[%d] fpid==[%d]\n",getpid(),getppid());
sleep(2);
return 9;
}
return 0;
}