子进程复制了父进程的什么

子进程通常复制父进程的虚拟地址空间,但并非物理内存。相同地址的不同值源于逻辑地址到物理地址的映射差异。这种现象称为写时复制,即在数据被修改时才会真正复制页面,有效降低系统开销。

每次看到多进程时,子进程复制了父进程的所有数据(代码段、数据段、BSS、堆、栈),我们想当然的认为:子进程开辟了一块新的空间,把父进程的所有数据都复制过来。

而且每次我们改变名字相同的变量的值,输出的变量的结果都是不一样的。这也就更加坚定了我们的信心。但是,事实真的是这样吗?

运行下面这段代码,你会发现父子进程相同的变量输出的值不一样,但是地址却是一样的,看下面代码:

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>

main
### 子进程父进程PID的关系及原理 在 Linux 系统中,`fork()` 系统调用用于创建一个新的进程(即子进程),该子进程几乎完全复制父进程的状态。然而,在实际操作中,子进程父进程之间存在一些重要的区别。 #### 1. **PID 的定义** 每个进程都拥有唯一的标识符 `PID`,它类似于进程的“身份证”。无论是系统级进程还是用户启动的进程,其 `PID` 均由操作系统分配并保证全局唯一性[^2]。当通过 `fork()` 创建子进程时,尽管子进程继承了父进程的大部分资源(如文件描述符、环境变量等),但它会被赋予一个全新的 `PID` 来区分于父进程和其他现有进程。 #### 2. **`fork()` 返回的意义** - 在父进程中,`fork()` 返回的是新创建的子进程的 `PID`。 - 在子进程中,`fork()` 返回为 `0`,表示这是子进程本身[^4]。 因此,可以通过检测 `fork()` 的返回来判断当前代码是在父进程中执行还是在子进程中执行。 #### 3. **子进程父进程 PID 的具体关系** 从引用的内容可以看出,子进程的实际 `PID` 并不会等于 `0`。实际上,子进程的 `PID` 是由操作系统动态分配的一个正整数,并且通常情况下,子进程的 `PID` 和父进程的 `PID` 不相同[^5]。例如: ```python import os pid = os.fork() if pid < 0: print("Create process failed") elif pid == 0: # 子进程逻辑 print(f"子进程的PID: {os.getpid()}") # 获取当前子进程的PID print(f"从子进程中获取父进程的PID: {os.getppid()}") # 获取父进程的PID else: # 父进程逻辑 print(f"在父进程中获取子进程的PID: {pid}") # fork() 返回子进程的PID print(f"获取父进程的PID: {os.getpid()}") # 当前父进程的PID ``` 运行结果可能如下所示: ``` 在父进程中获取子进程的PID: 5363 获取父进程的PID: 5362 子进程的PID: 5363 从子进程中获取父进程的PID: 5362 ``` 可以看到,父进程的 `PID` 为 `5362`,而子进程的 `PID` 则被分配为 `5363`。这表明两者具有不同的 `PID`,从而能够相互独立地运行[^3]。 #### 4. **特殊情况下的讨论** 理论上讲,只有在同一内核上下文中可能存在某些特殊场景下两个实体共享相同的 `PID`,但这仅限于内核线程内部实现细节,并不适用于普通的用户态进程或线程。对于标准意义上的父子进程而言,它们始终具备互异的 `PID`。 --- ###
评论 2
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值