Fork() 函数:“父” 与 “子” 进程的交互(进程的创建)
作者:mmseoamin日期:2023-12-14

Fork() 函数:“父” 与 “子” 进程的交互(进程的创建),在这里插入图片描述,第1张

阅读导航

  • 前言
  • 一、fork函数初识
    • 1. 基本概念
    • 2. fork函数返回值
    • 二、fork函数的写时拷贝
    • 三、总结
    • 温馨提示

      前言

      前面我们讲了C语言的基础知识,也了解了一些数据结构,并且讲了有关C++的一些知识,也学习了一些Linux的基本操作,也了解并学习了有关Linux开发工具vim 、gcc/g++ 使用、yum工具以及git 命令行提交代码也相信大家都掌握的不错,上一篇文章我们了解了关于进程的基本概念,今天博主带大家了解一下编程中的一个非常重要的函数 —— fork(), 下面话不多说坐稳扶好咱们要开车了!!!😍

      一、fork函数初识

      1. 基本概念

      fork函数是操作系统中的一个系统调用,用于创建一个新的进程,该进程是调用fork函数的进程的一个副本。新创建的进程称为子进程,原始进程称为父进程。

      fork函数的函数原型:

      #include 
      pid_t fork(void);
      

      2. fork函数返回值

      1. 父进程中的返回值:

        • 如果fork函数返回一个大于0的值,表示当前执行的是父进程。这个返回值是子进程的PID(进程ID),可以用来操作子进程。
        • 如果fork函数返回-1,表示创建子进程失败,通常是因为系统资源不足或权限不够等原因,此时应该处理错误情况。
        • 子进程中的返回值:

          • 如果fork函数返回0,表示当前执行的是子进程。可以根据需要在子进程中执行相应的任务逻辑。

      根据fork函数的返回值,可以在程序中使用条件语句来区分父进程和子进程的不同逻辑,从而实现不同的处理方式。例如,可以在父进程中等待子进程的完成,或者在子进程中执行某种特定的任务。

      #include 
      #include 
      #include 
      int main() {
          pid_t pid = fork();
          if (pid > 0) {
              // 父进程逻辑
              printf("This is the parent process. Child's PID: %d\n", pid);
          } else if (pid == 0) {
              // 子进程逻辑
              printf("This is the child process. Parent's PID: %d\n", getppid());
          } else {
              // fork失败
              fprintf(stderr, "Failed to create child process.\n");
              return 1;
          }
          // 父子进程共享的代码
          printf("This message is printed by both parent and child processes.\n");
          return 0;
      }
      

      需要注意的是,fork函数的使用可能会导致代码的分支,需要小心处理父子进程之间共享的资源以及避免产生竞争条件,以确保程序的正确性和可靠性。

      二、fork函数的写时拷贝

      fork函数的写时拷贝(Copy-on-Write,COW)是一种优化策略,用于在创建子进程时避免立即复制父进程的整个地址空间,这种机制可以提高性能和减少内存消耗。

      在传统的fork操作中,父进程会创建一个子进程,并且子进程会复制父进程的所有资源,包括内存空间、文件描述符等。这样的完全复制操作非常消耗时间和内存。而使用写时拷贝机制,只有在需要修改共享的内存页时才会进行复制操作,从而节省了系统资源。

      具体来说,当调用fork函数创建子进程时,操作系统会执行以下步骤:

      1. 父进程会创建一个与自己拥有相同地址空间的子进程。
      2. 子进程继承了父进程的页表,这意味着它与父进程共享相同的虚拟内存地址空间。
      3. 在初始阶段,父进程和子进程共享所有的物理页面,这些页面被标记为“只读”。
      4. 当父进程或子进程尝试修改共享的内存页时,操作系统会将相应的页面复制到一个新的物理页面,并将其标记为“可写”。
      5. 父进程和子进程现在各自拥有一个独立的物理页面,它们不再共享相同的数据。

      通过写时拷贝技术,父进程和子进程共享大部分内存页,只在需要修改共享内存时才进行复制操作。这样可以节省时间和内存,并提高系统性能。例如,在fork之后,如果子进程立即执行exec函数加载了一个新的程序,那么就不需要进行任何复制操作,这是因为子进程并不需要修改父进程的内存数据。

      Fork() 函数:“父” 与 “子” 进程的交互(进程的创建),在这里插入图片描述,第2张

      需要注意的是,写时拷贝只是在逻辑上实现了共享,而不是物理上的共享。父进程和子进程仍然拥有各自独立的虚拟地址空间,它们之间的共享是通过允许读取相同的物理内存来实现的,只有在修改时才会发生内存复制。

      总结起来,fork函数的写时拷贝机制使得父进程和子进程在初始阶段共享相同的内存空间,只有在需要修改共享内存时才进行复制操作,从而提高了性能和降低了资源消耗。

      三、总结

      我们首先了解了fork函数的基本概念。fork函数是操作系统中的一个系统调用,用于创建一个新的子进程。父进程调用fork函数后,会创建一个与自己拥有相同地址空间的子进程,这包括了代码、数据、堆栈等。子进程是通过复制父进程的地址空间来实现的。我们还学习了fork函数的返回值。fork函数在父进程中返回子进程的进程ID(PID),而在子进程中返回0。通过这个返回值,我们可以在父子进程中进行不同的处理逻辑。

      在第二部分中,我们介绍了fork函数的写时拷贝(Copy-on-Write,COW)机制。传统的fork操作会完全复制父进程的内存空间,这在资源消耗和性能方面可能存在问题。而使用写时拷贝机制,只有在父进程或子进程尝试修改共享的内存页时才进行复制操作,从而减少复制的次数和消耗的资源。通过写时拷贝,父进程和子进程共享大部分内存页,只有在需要修改共享内存时才进行复制操作。这种优化策略提高了性能,并减少了内存资源的消耗。需要注意的是,写时拷贝只是逻辑上的共享,父进程和子进程仍然拥有各自独立的虚拟地址空间。

      这些知识有助于理解fork函数的工作原理和使用方式,以及在编写多进程程序时进行性能优化的思路。

      温馨提示

      感谢您对博主文章的关注与支持!如果您喜欢这篇文章,可以点赞、评论和分享给您的同学,这将对我提供巨大的鼓励和支持。另外,我计划在未来的更新中持续探讨与本文相关的内容。我会为您带来更多关于Linux以及C++编程技术问题的深入解析、应用案例和趣味玩法等。如果感兴趣的话可以关注博主的更新,不要错过任何精彩内容!

      再次感谢您的支持和关注。我们期待与您建立更紧密的互动,共同探索Linux、C++、算法和编程的奥秘。祝您生活愉快,排便顺畅!

      Fork() 函数:“父” 与 “子” 进程的交互(进程的创建),在这里插入图片描述,第3张