当一个进程的父进程退出时,它的所有子进程变为孤儿进程,那么这些孤儿进程的父进程变成谁了呢?一般来讲,init进程会收养这些进程,完成对他们的状态收集工作。在《linux内核设计与实现》书中写到“给子进程在当前线程组内找一个线程作为父亲,如果不行,就让init做父进程”。
以下是测试的代码:
#define _GNU_SOURCE
#include"stdio.h"
#include <sys/types.h>
#include <dirent.h>
#include <string.h>
#include "sched.h"
#include "signal.h"
void createSunofClone();
/*
创建进程树,让其他进程睡眠10妙,中间被删除的进程睡眠5秒后自杀,查看被删除后,进程树的结构
*/
int main(){
pid_t pid,pid1;
void * stack;
stack = (void*)malloc(8192);
/*
if(!(pid1=fork())){
printf("sun--1.1: pid=%d,fpid=%d\n",getpid(),getppid());
sleep(10);
printf("sun--1.1: pid=%d,fpid=%d\n",getpid(),getppid());
exit(0);
}*/
if(!(pid=fork())){
printf("sun--1: pid=%d,fpid=%d\n",getpid(),getppid());
pid_t pid2;
pid_t pid22;
pid_t pid23;
/*
if(!(clone(&createSunofClone,(void*)stack+16385,CLONE_VM|CLONE_FILES,0))){
}
if(!(pid22=fork())){
printf("sun--2.2: pid=%d,fpid=%d\n",getpid(),getppid());
sleep(10);
printf("sun--2.2: pid=%d,fpid=%d\n",getpid(),getppid());
exit(0);
}
if(!(pid23=fork())){
printf("sun--2.3: pid=%d,fpid=%d\n",getpid(),getppid());
sleep(10);
printf("sun--2.3: pid=%d,fpid=%d\n",getpid(),getppid());
exit(0);
}*/
if(!(pid2=fork())){
printf("sun--2: pid=%d,fpid=%d\n",getpid(),getppid());
pid_t pid3;
if(!(pid=fork())){
printf("sun--3: pid=%d,fpid=%d\n",getpid(),getppid());
sleep(10);
printf("sun--3: pid=%d,fpid=%d\n",getpid(),getppid());
}
else{
// wait();
printf("praent--3: pid=%d,fpid=%d\n",getpid(),getppid());
sleep(5);
printf("-----------删除进程-------------\n");
exit(0);
}
}else{
// wait();
printf("praent--2: pid=%d,fpid=%d\n",getpid(),getppid());
sleep(10);
printf("praent--2: pid=%d,fpid=%d\n",getpid(),getppid());
}
}else{
printf("praent--1: pid=%d,fpid=%d\n",getpid(),getppid());
sleep(15);
printf("praent--1:id=%d,fpid=%d\n",getpid(),getppid());
}
}
void createSunofClone(){
printf("clone--sun--2.2: pid=%d,fpid=%d\n",getpid(),getppid());
sleep(10);
printf("clone--sun--2.2: pid=%d,fpid=%d\n",getpid(),getppid());
exit(0);
}
1.第一种情况:
执行上面的代码,结果如下:
进程树的情况为
变为
这里1817是一个叫user-init的进程,可以说是init的一个替代品。
这种情况下,孤儿进程被init进程收养。
2.第二种情况:
把原来代码上的部分注释代码加进来(读者可根据上述代码自己增删),结果如下:
进程树的情况为:
变为
同样,孤儿进程被init进程收养。
3.第三种情况:
继续修改代码,结果如下:
进程树的情况为:
变为
孤儿进程还是被init收养。
利用clone代替fork执行也是同样的情况。
4.结果
不管是哪种情况,孤儿进程最终都是被init收养,并没有发现某个孤儿进程被其他进程收养。