fork 复制进程
进程概论
(1) 进程:一个正在运行的程序。
(2) PCB : 即是进程控制块,是进程存在的唯一标志。用来描述进程的属性信息。
fork 函数会复制此进程的PCB块,新生成一个进程,调用 fork 函数的进程为父进程,新生成的进程为子进程。
在父进程中返回子进程的 pid,在子进程中返回 0,失败返回-1。
测试代码:
getpid()获取当前进程的pid
getppid()获取当前进程的父进程
程序运行的 fork()处,会将父进程整体复制一份,子进程只执行fork()后的语句,不执行fork()以前的语句。
父子进程并发运行的理解
逻辑地址 物理地址
测试代码
运行结果:
如上图所示:子进程的n 和 父进程的n 值不同,但是他们的地址相同,这是因为他们使用的是逻辑地址,即偏移量
如图
这很好理解,比如你的座位号是 10号,不论在哪个教室,你的座位号都是10号。
写时拷贝
传统的fork()系统调用直接把所有的资源复制给新创建的进程。这种实现过于简单并且效率低下,因为它拷贝的数据也许并不共享,更糟的情况是,如果新进程打算立即执行一个新的映像那么所有的拷贝都将前功尽弃。Linux的fork()使用写时拷贝 (copy-on-write)页实现。写时拷贝是一种可以推迟甚至免除拷贝数据的技术。内核此时并不复制整个进程地址空间,而是让父进程和子进程共享同一个拷贝。只有在需要写入的时候,数据才会被复制,从而使各个进程拥有各自的拷贝。也就是说,资源的复制只有在需要写入的时候才进行,在此之前,只是以只读方式共享这种技术使地址空间上的页的拷贝被推迟到实际发生写人的时候。