fork, vfork and write-on-copy

本文探讨了在创建子进程时使用的两种高效技术:写时复制和Vfork。写时复制通过仅复制被修改的内存页来节省资源,而Vfork则让子进程直接使用父进程的地址空间,适用于子进程创建后立即执行exec的情况。

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

Fork

传统方式下,Fork创建一个子进程,并为子进程创建一个父进程地址空间的拷贝。然而,由于许多子进程在创建之后通常马上会执行系统调用exec,所以父进程地址空间的复制可能没有必要,从而造成效率和内存的极大浪费。因此,就产生了一种称为“写时复制”的技术。

写时复制

写时复制允许子进程与父进程在开始时共享同一页面。但这些页面被标记为“写时复制”,即如果任何一个进程需要对页进行写操作,就会创建这个共享页的拷贝。例如,假设子进程试图修改含有部分栈的页,且操作系统能够识别出该页为写时复制页,则操作系统就会创建该页的一个拷贝,并将它映射的子进程的地址空间里。这样,子进程修改的就是其复制的页,而不是父进程的页。采用写时复制技术,只有被进程所修改的页才会复制,而所有非修改的页可为父进程和子进程共享。注意,并不是所有的页都标记为写时复制,只有可能修改的页才需要标记为写时复制,对不能修改的页,如代码页,可谓父进程和子进程所共享。

Vfork(Virtual memory fork)

Vfork不同于写时复制的fork。对于vfork,父进程会挂起,以确保子进程先运行。子进程使用父进程的地址空间。由于vfork不使用写时复制,因此如果子进程修改地址空间的任何页,这些修改在父进程重启时都是可见的。所以,vfork必须小心使用,以确保子进程不会修改父进程的地址空间。Vfork主要用于在进程创建后立即调用exec的情况,这样既没有出现复制页,也不会修改父进程的地址空间,所以是一种很有效的进程创建方法。

 

在Linux系统中,`fork()``vfork()`是两个用于创建新进程的系统调用。它们之间存在显著的区别,这些区别影响着它们的行为、性能以及适用场景。 ### `fork()`函数 `fork()`函数通过复制当前进程来创建一个新的子进程。这个新进程几乎是原始进程的一个副本,包括代码段、数据段、堆栈等。然而,现代Linux内核采用了写时复制(Copy-on-Write, CoW)技术[^2],这意味着实际的数据复制只有当父子进程中任一进程试图修改内存时才会发生。这种优化减少了不必要的资源消耗,并提高了效率。 - **行为**:子进程获得父进程的所有内存页面的拷贝,但仅在需要时才进行深拷贝。 - **独立性**:子进程与父进程完全隔离,拥有自己的地址空间。 - **执行顺序**:无法保证哪个进程先运行;通常由操作系统调度决定。 - **使用场景**:适合于那些需要完全独立的新进程的情况,尤其是在后续会调用`exec`系列函数加载新的程序到子进程中去的情形。 ### `vfork()`函数 相比之下,`vfork()`的设计目的是为了提高某些特定情况下的效率。它允许子进程共享父进程的地址空间直到子进程调用了`_exit()`或其中一个`exec`家族函数[^3]。在此期间,如果子进程尝试对内存做出任何更改,则可能导致未定义的行为。 - **行为**:子进程不复制父进程的页表项而是直接使用父进程的,因此非常快。 - **独立性**:子进程父进程共享相同的地址空间,这要求开发者特别小心处理内存操作。 - **执行顺序**:传统上,`vfork()`会挂起父进程直至子进程结束或者调用`exec`/`_exit`,确保子进程优先执行。 - **使用场景**:适用于快速产生一个仅用来执行新程序的子进程的情况,且不需要访问除其自身栈之外的其他内存区域。 ### 应用实例对比 #### 使用`fork()` ```c #include <stdio.h> #include <unistd.h> #include <stdlib.h> int main() { pid_t pid = fork(); if (pid < 0) { perror("fork"); exit(EXIT_FAILURE); } else if (pid == 0) { // 子进程 printf("Child process via fork.\n"); } else { // 父进程 printf("Parent process after fork.\n"); } return 0; } ``` #### 使用`vfork()` ```c #include <stdio.h> #include <unistd.h> #include <stdlib.h> int main() { pid_t pid = vfork(); if (pid < 0) { perror("vfork"); exit(EXIT_FAILURE); } else if (pid == 0) { // 子进程 printf("Child process via vfork.\n"); _exit(EXIT_SUCCESS); // 必须使用_exit而不是exit以避免清理操作 } else { // 父进程 printf("Parent process after vfork.\n"); } return 0; } ``` ### 注意事项 尽管`vfork()`可能提供更好的性能,但由于它限制了子进程如何操作内存并且可能导致难以调试的问题,推荐尽可能使用`fork()`。除非是在对速度极为敏感的应用中,否则不应选择`vfork()`[^2]。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值