孤儿进程的父进程

当父进程退出,其子进程变为孤儿进程,由init进程接手。通过不同情况的代码测试,无论怎样,孤儿进程始终被init或其替代品user-init收养,负责收集子进程的状态。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

当一个进程的父进程退出时,它的所有子进程变为孤儿进程,那么这些孤儿进程的父进程变成谁了呢?一般来讲,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.第一种情况:

执行上面的代码,结果如下:
图1
进程树的情况为
变为
这里1817是一个叫user-init的进程,可以说是init的一个替代品。
这种情况下,孤儿进程被init进程收养。

2.第二种情况:

把原来代码上的部分注释代码加进来(读者可根据上述代码自己增删),结果如下:
图2
进程树的情况为:
变为
同样,孤儿进程被init进程收养。

3.第三种情况:

继续修改代码,结果如下:
图3
进程树的情况为:
变为
孤儿进程还是被init收养。
利用clone代替fork执行也是同样的情况。

4.结果

不管是哪种情况,孤儿进程最终都是被init收养,并没有发现某个孤儿进程被其他进程收养。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值