刚fork之后:
父子相同处: 全局变量、.data、.text、栈、堆、环境变量、用户ID、宿主目录、进程工作目录、信号处理方式...
父子不同处: 1.进程ID 2.fork返回值 3.父进程ID 4.进程运行时间 5.闹钟(定时器) 6.未决信号集
似乎,子进程复制了父进程0-3G用户空间内容,以及父进程的PCB,但pid不同。真的每fork一个子进程都要将父进程的0-3G地址空间完全拷贝一份,然后在映射至物理内存吗?
当然不是!父子进程间遵循读时共享写时复制的原则。这样设计,无论子进程执行父进程的逻辑还是执行自己的逻辑都能节省内存开销。
下面一段简单的程序可以很好的测试出父子进程是否共享全局变量。
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int a = 100; //.data
int main(void)
{
pid_t pid;
pid = fork();
if(pid == 0){ //son
a = 2000;
printf("child, a = %d\n", a);
} else {
sleep(1); //保证son先运行
printf("parent, a = %d\n", a);
}
return 0;
}
重点注意!可以看出父子进程并不是共享全局变量
父子进程共享的有:
1. 文件描述符(打开文件的结构体)
2. mmap建立的映射区
特别的,fork之后父进程先执行还是子进程先执行不确定。取决于内核所使用的调度算法。