【Linux】fork和vfork

本文详细介绍了fork和vfork函数的功能及区别。fork函数创建一个新进程并完全复制父进程的数据段和代码段;而vfork函数创建的新进程与父进程共享数据段,适用于立即调用exec的情况。此外,文章还解释了两种函数的返回值差异以及它们如何影响进程间的执行顺序。

 一、fork函数
  函数原型:
这里写图片描述
  函数功能:
  一个函数可以调用fork函数创建一个新进程。
  函数返回值:
由fork函数创建的新进程叫做子进程,fork函数被调用一次但是返回两次,两次返回的唯一区别是子进程的返回值是0,父进程的返回值是新进程的进程ID。
  将子进程进程ID返回给父进程的原因是:因为一个进程的子进程可能有多个,并且没有一个函数可以获得其所有子进程的进程ID。
  子进程返回0的原因是:一个进程只会有一个父进程,所以子进程总是可以调用getppid获得其父进程的进程ID(进程ID 0总是由内核交换进程使用,所以一个子进程的进程ID不可能为0)
  父进程和子进程继续执行fork之后的指令。子进程是父进程的副本。例如子进程获得父进程数据空间,堆栈的副本。这是副本,父子进程并不共享这些存储空间部分,父子进程共享正文段。
  由于在fork之后经常跟着exec,所以现在的很多实现并不执行父进程数据段、栈和堆的完全复制。作为替代使用了写时拷贝技术,这些区域由父子进程共享,内核将其权限改成只读的当、子进程中的某个想要去修改这些区域的时候,内核只为修改的区域创建一个副本。
  一般来说fork之后是先执行父进程还是先执行子进程是不确定的,取决于内二所使用的调度算法,要想实现父、子进程之间的同步,则要求某种形式的进程间通信。
  文件共享:
当父进程的标准输出重定向时,子进程的标准也应该重定向,fork的一个特性是,父进程的所有打开文件描述符都被复制到子进程中。
  在fork之后处理文件描述符一般有两种常见情况:
(1)父进程等待子进程完成
(2)父子进程各自执行不同的程序段
这里写图片描述
  父子进程之间的区别是:
fork的返回值不同
进程ID不同
两个进程具有不同的父进程ID:子进程的父进程ID是创建它的进程的进程ID,父进程的父进程ID不变
父进程设置的文件锁不会被子进程继承
子进程的未处理的的闹钟被清除
子进程的未处理信号集设置为空集
  二、vfork函数
  函数原型:
这里写图片描述
  函数功能:
用于创建一个新进程,而该进程的目的是exec一个程序,vfork和fork函数一样都创建一个新进程,但是它并不将父进程的地址空间完全复制到子进程中,因为子进程会立即调用exec(或exit)于是就不会访存该空间,相反,在子进程调用exec或者exit之前,它会在父进程的空间中运行,提高了效率。
  fork和vfork之间的另一个区别是:vfork保证子进程先运行,在它调用exec或者exit之后父进程才可能被调度运行,(如果调用这两个函数之前子进程依赖于父进程的进一步动作,那么可能会导致死锁)
  函数返回值:vfork函数的返回值和fork函数相同,但是语义不同
总结一下这fork和vfork之间的区别:
1、fork()子进程拷贝父进程的数据段、代码段
vfork()子进程和父进程共享数据段
2、fork()父子进程的执行次序不确定
vfork()保证子进程先运行,在调用exec或者exit函数之前,和父进程数据是共享的,在它调用exec或者exit之后父进程才可能被调度运行
3、vfork()保证子进程先被运行,在它调用exec或者exit之后父进程才会被调度运行,如果调用这两个函数之前,子进程依赖于父进程的进一步动作,则会导致死锁。

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、付费专栏及课程。

余额充值