fork vfork

vfork

  
   vfork(建立一个新的进程)
  相关函数 wait,execve
  头文件 #include<unistd.h>
  定义函数 pid_t fork(void);
   函数说明
  vfork()会产生一个新的子进程.但是vfork创建的子进程与父进程共享数据段,而且由vfork创建的
  子进程将先于父进程运行.fork()的使用详见百度词条fork().
  vfork()用法与fork()相似.但是也有区别,具体区别归结为以下3点:
  1. fork():子进程拷贝父进程的 数据段,代码段. vfork():子进程与父进程共享数据段.
  2. fork():父子进程的执行次序不确定.
  vfork():保证子进程先运行,在调用exec或exit之前与父进程数据是共享的,在它调用exec
  或exit之后父进程才可能被调度运行。
  3. vfork()保证子进程先运行,在她调用exec或exit之后父进程才可能被调度运行。如果在
  调用这两个 函数之前子进程依赖于父进程的进一步动作,则会导致死锁。
  下面通过几个例子加以说明:
   返回值
  如果vfork()成功则在父进程会返回新建立的子进程代码(PID),
  而在新建立的子进程中则返回0。如果vfork失败则直接返回-1,失
  败原因存于errno中。
   错误代码
  EAGAIN 进程数已达系统规定上限
  ENOMEM 内存不足,无法配置核心所需的数据结构空间。
   范例
  #include<unistd.h>
  main()
  {
  pid_t pid;
  if((pid = vfork()) < 0){
  perror("vfork() failed");
  eixt(1);
  }else if(pid == 0){
  printf(“This is the child process\n”);
  }else{
  printf(“This is the parent process\n”);
  }
  }
   执行
  this is the child process
  this is the parent process

fork

目录

fork()函数 Linux
中文翻译

编辑本段fork()函数 Linux

  # include<unistd.h>
  # include<sys/types.h>
  函数定义:
   pid_t forkvoid);
  (pid_t 是一个宏定义,其实质是int 被定义在# include< sys/types.h>中)
  返回值: 若成功调用一次则返回两个值,子进程返回0,父进程返回子进程ID;否则,出错返回-1
  函数说明:
  一个现有进程可以调用fork函数创建一个新进程。由fork创建的新进程被称为子进程(child process)。fork函数被调用一次但返回两次。两次返回的唯一区别是子进程中返回0值而父进程中返回子进程ID。
  子进程是父进程的副本,它将获得父进程数据空间、堆、栈等资源的副本。注意,子进程持有的是上述存储空间的“副本”,这意味着父子进程间不共享这些存储空间。
  linux将复制父进程的地址空间内容给子进程,因此,子进程有了独立的地址空间。
   为什么fork会返回两次?
  由于在复制时复制了父进程的堆栈段,所以两个进程都停留在fork函数中,等待返回。因为fork函数会返回两次,一次是在父进程中返回,另一次是在子进程中返回,这两次的返回值是不一样的。过程如下图
   调用fork之后,代码有两份,都从fork函数中返回,箭头表示各自的执行处
  示例代码:
  # include< sys/types.h> //对于此程序而言此头文件用不到
  # include< unistd.h>
  # include< stdio.h>
   int main( int argc,  char ** argv )
  {
   int pid = fork();
   if (pid < 0) {
  // printf("error!");
  }  else if( pid == 0 ) {
  // printf("This is the child process!");
  }  else{
  // printf("This is the parent process! child process id = %d", pid);
  }
   return 0;
  }
  fork()系统在Linux中的返回值是没有NULL的.
  Error Codes
  出错返回错误信息如下:
   EAGAIN
  达到进程数上限.
   ENOMEM
  没有足够空间给一个新进程分配.



### forkvfork 的执行顺序差异 在 Linux 系统中,`fork()` 和 `vfork()` 是用于创建新进程的两个重要系统调用。尽管两者都实现了进程复制的功能,但它们在执行顺序上的表现有显著的不同。 #### 1. **`fork()` 的执行顺序** 当通过 `fork()` 创建一个新的子进程时,父进程和子进程会分别获得独立的内存副本[^1]。这意味着父子进程之间不会共享地址空间。由于这种特性,在 `fork()` 调用完成后,父进程和子进程都会继续执行 `fork()` 后面的代码,但是具体谁先运行是由操作系统的调度策略决定的[^5]。因此: - 父进程和子进程的执行顺序是不确定的。 - 这种不确定性使得开发者无法预测哪一个进程会在另一个之前运行。 示例代码如下: ```c #include <stdio.h> #include <unistd.h> int main() { printf("Before fork.\n"); int ret = fork(); if (ret == 0) { // 子进程 printf("Child process running\n"); } else if (ret > 0) { // 父进程 printf("Parent process running\n"); } return 0; } ``` 在这个例子中,“Child process running” 或 “Parent process running” 可能会交替出现,因为两者的执行顺序由内核调度器控制。 --- #### 2. **`vfork()` 的执行顺序** 相比之下,`vfork()` 提供了一种更高效的进程创建方法。它并不会像 `fork()` 那样为子进程分配全新的虚拟地址空间,而是让子进程暂时借用父进程的地址空间[^4]。为了防止数据损坏,`vfork()` 强制规定只有当子进程显式调用了 `exec*()` 家族函数或者退出 (`_exit`) 之后,父进程才会恢复执行。 换句话说: - 使用 `vfork()` 时,总是保证子进程优先于父进程运行。 - 如果子进程中未及时调用 `exec*()` 函数或 `_exit()`,则可能导致死锁情况发生。 以下是基于 `vfork()` 的简单演示代码片段: ```c #include <stdio.h> #include <unistd.h> int main() { printf("Before vfork.\n"); int ret = vfork(); // 注意这里使用的是 vfork() if (ret == 0) { // 子进程 printf("Child process running and will exit now.\n"); _exit(0); // 必须立即终止子进程 } else if (ret > 0) { // 父进程 printf("Parent process resumes after child exits or execs.\n"); } return 0; } ``` 在此情况下,“Child process running...” 总是在 “Parent process resumes...” 输出前完成打印并结束其生命周期。 --- ### 结论 综上所述,`fork()` 和 `vfork()` 在执行顺序方面的核心区别在于: - 对于 `fork()`,父进程和子进程可以任意顺序运行; - 对于 `vfork()`,子进程始终优先运行,并且直到子进程调用特定函数(如 `exec*()` 或 `_exit()`)后,父进程才能重新获取 CPU 控制权。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值