进程的创建

关于进程创建的一些细节都在注释中

cpp 复制代码
/*
    #include <sys/types.h>
    #include <unistd.h>

    pid_t fork(void);
        函数的作用:用于创建子进程。
        返回值:
            fork()的返回值会返回两次。一次是在父进程中,一次是在子进程中。
            在父进程中返回创建的子进程的ID,
            在子进程中返回0
            如何区分父进程和子进程:通过fork的返回值。
            在父进程中返回-1,表示创建子进程失败,并且设置errno
        失败的两个主要原因:
            当前系统的进程数已经达到了系统规定的上限,这时 errno 的值被设置为 EAGAIN
            系统内存不足,这时 errno 的值被设置为 ENOMEM

        父子进程之间的关系:
        区别:
            1.fork()函数的返回值不同
                父进程中: >0 返回的子进程的ID
                子进程中: =0
            2.pcb中的一些数据
                当前的进程的id pid
                当前的进程的父进程的id ppid
                信号集

        共同点:
            某些状态下:子进程刚被创建出来,还没有执行任何的写数据的操作
                - 用户区的数据
                - 文件描述符表
        
        父子进程对变量是不是共享的?
            - 刚开始的时候,是一样的,共享的。如果修改了数据,不共享了。
            - 读时共享(子进程被创建,两个进程没有做任何的写的操作),写时拷贝。
        
*/
#include <sys/types.h>
#include <unistd.h>
#include <stdio.h>

int main() {

    int num = 10;

    // 创建子进程
    pid_t pid = fork();// pid底层就是int类型
    // fork()以后,子进程的用户区数据和父进程一样。内核区也会拷贝过来,但是父子进程的pid不一样

    // 判断是父进程还是子进程
    if(pid > 0) {
        // printf("pid : %d\n", pid);
        // 如果大于0,返回的是创建的子进程的进程号,当前是父进程
        printf("i am parent process, pid : %d, ppid : %d\n", getpid(), getppid());

        printf("parent num : %d\n", num);
        num += 10;
        printf("parent num += 10 : %d\n", num);

    } else if(pid == 0) {
        // 当前是子进程
        printf("i am child process, pid : %d, ppid : %d\n", getpid(), getppid());
       
        printf("child num : %d\n", num);
        num += 100;
        printf("child num += 100 : %d\n", num);
    }

    // for循环
    for(int i = 0; i < 3; i++) {
        printf("i : %d , pid : %d\n", i , getpid());
        sleep(1);
    }

    return 0;
}

/*
    实际上,更准确来说,Linux 的 fork() 使用是通过写时拷贝 (copy- on-write) 实现。
    写时拷贝是一种可以推迟甚至避免拷贝数据的技术。
    内核此时并不复制整个进程的地址空间,而是让父子进程共享同一个地址空间。
    只用在需要写入的时候才会复制地址空间,从而使各个进行拥有各自的地址空间。
    也就是说,资源的复制是在需要写入的时候才会进行,在此之前,只有以只读方式共享。
    注意:fork之后父子进程共享文件(将父进程的整个地址空间复制给子进程,包括文件描述符。),
    fork产生的子进程与父进程相同的文件文件描述符指向相同的文件表,引用计数增加,共享文件偏移指针。
*/


/*  
    文件描述符的细节:
    在单个进程中,文件描述符的值是唯一的,不会重复。每个文件描述符都对应于进程的文件描述符表中的唯一条目。
    但在多进程情况下,由于 fork 调用会复制父进程的文件描述符表给子进程,导致子进程获得了与父进程相同的文件描述符值。
    这样,不同的进程可以有相同的文件描述符值,但它们对应的是各自独立的文件描述符表中的不同条目。
*/
相关推荐
tan77º28 分钟前
【项目】分布式Json-RPC框架 - 项目介绍与前置知识准备
linux·网络·分布式·网络协议·tcp/ip·rpc·json
快乐zbc32 分钟前
数学建模Topsis法笔记
笔记·数学建模
yiqiqukanhaiba32 分钟前
STM32学习笔记14-I2C硬件控制
笔记·stm32·学习
悠哉悠哉愿意1 小时前
【Python语法基础学习笔记】if语句
笔记·python·学习
岑梓铭2 小时前
考研408《计算机组成原理》复习笔记,第五章(1)——CPU功能和结构
笔记·考研·408·计算机组成原理·计组
正在努力的小河3 小时前
Linux设备树简介
linux·运维·服务器
荣光波比3 小时前
Linux(十一)——LVM磁盘配额整理
linux·运维·云计算
LLLLYYYRRRRRTT4 小时前
WordPress (LNMP 架构) 一键部署 Playbook
linux·架构·ansible·mariadb
轻松Ai享生活4 小时前
crash 进程分析流程图
linux
universe_015 小时前
day25|学习前端js
前端·笔记