【Linux】fork()

目录

1、fork是什么

2、 fork复制原理

3、逻辑地址与物理地址

4、计算fork()输出次数


1、fork是什么

  • linux下创建新进程的系统调用的是fork(),其定义如下:
  • ​​​​​​​#include<sys/types.h>
    #include<unistd.h>
    pid_t fork(void);
  • fork函数的功能就是复制当前进程,在内核进程表中创建一个新的进程表象,该进程称为子进程,被复制的进程称为父进程
  • 子进程的代码和父进程的完全相同,同时它还会复制父进程的数据(堆数据、栈数据和静态数据)。数据的复制采用的是写时复制。此外,创建子进程后,父进程打开的文件描述符默认在子进程中也是打开的,且文件描述符的引用计数+1。

2、 fork复制原理

  • 父进程有pcb,父进程内调用fork()函数,fork复制一份父进程的pcb,然后申请一个PID,把子进程的pcb里的pid的加1,父进程的返回值为子进程的pid,子进程的返回值为0,如下图所示:
  • bf2ba936cfb649ab8c5e0606fabd3af1.png
  • 总结:
    • 子进程的PID=父进程的PID+1
    • 父进程的返回值为子进程的PID,子进程的返回值为0。该返回值是后续代码判断当前进程是父进程还是子进程的依据。fork调用失败返回-1,并设置errno

3、逻辑地址与物理地址

  • 父进程fork后产生的子进程,逻辑地址一致,物理地址不一致
  • 在进程中看到的地址都是逻辑地址
  • 进程有自己的独立地址空间,每启动一个进程,系统就会为它分配地址空间,分配的是逻辑地址。要使用地址时再去寻找空闲的地址——物理地址。
  • 有些时候,父子进程的某个变量的输出地址可能相同,只可能是其逻辑地址相同,要想知道它们在内存中的物理地址,就必须根据逻辑地址判断页表号(如下图所示),再根据各自的页表进行映射,找到其在内存中的具体位置——物理地址。不同进程的逻辑地址没有可比性,但是如果在同一个进程中,若地址相同,那就证明其在同一段物理内存中。

4、计算fork()输出次数

例题分析——打印几个A?

  • 解:
  •  去掉"\n"

  • 有\n 直接打印
  • 没有\n 放在缓冲区内

  • 在该例中,输出结果为BAA
  • 分析:
    • main函数内第一句,把A放到缓冲区,然后打印B, fork又把复制一次,两个缓冲区的A都被打印出来,因此输出结果依次为BAA
  •  
  •  

### Linux 中 `fork` 函数的使用说明 #### 一、函数定义与返回值解释 `fork()` 是 Linux 系统中的一个重要系统调用,其功能是创建一个新的子进程。新创建的子进程几乎完全复制父进程的状态,包括内存空间、文件描述符等[^3]。 该函数的声明如下所示: ```c #include <sys/types.h> #include <unistd.h> pid_t fork(void); ``` 当成功调用时,`fork()` 的行为取决于当前运行的是哪个进程: - **对于父进程**:`fork()` 返回正值,表示新创建的子进程 ID (PID)[^4]。 - **对于子进程**:`fork()` 返回 0,表明这是由 `fork()` 创建的新进程实例。 - 如果发生错误,则会向父进程返回负值 `-1`,并设置全局变量 `errno` 来指示具体原因。 --- #### 二、典型的应用场景 `fork()` 常被用来实现多任务处理或者构建服务端程序,在这些情况下它能够有效地支持并发操作和资源隔离需求: 1. 并发执行多个独立的任务; 2. 启动守护进程或将某些长期运行的服务置于后台工作模式下; 3. 实现父子进程之间的基本消息传递机制——尽管更复杂的 IPC 需要额外的技术手段来完成[^2]。 下面通过几个具体的例子进一步阐述如何实际运用此 API 完成上述目标之一即简单的并发计算任务分配给不同的 CPU 核心去完成从而提高整体效率。 --- #### 三、代码示例 这里给出一段完整的 C 语言源码展示怎样利用 `fork()` 构建两个分支路径分别打印各自的信息直到结束为止: ```c #include <stdio.h> #include <stdlib.h> #include <unistd.h> /* For fork() */ #include <sys/types.h> /* For pid_t */ int main(){ pid_t child_pid; printf("Parent process starting...\n"); // Create a new child process. child_pid = fork(); if(child_pid == -1){ perror("Failed to create child"); exit(EXIT_FAILURE); } if(0 == child_pid){ // Child Process Block printf("Child PID=%d, Parent PID=%d\n", getpid(), getppid()); sleep(2); // Simulate some work done by the child. printf("This is output from CHILD PROCESS.\n"); }else{ // Parent Process Block printf("Child PID=%d created successfully!\n",child_pid); sleep(1); // Let parent do something before waiting on its children. printf("Waiting for child process %d to finish...",child_pid); wait(NULL); // Wait until any one of my direct descendants terminates. printf("\t...done! All processes have finished now.\n"); } return EXIT_SUCCESS; } ``` 在这个例子中可以看到我们先检查了 `fork()` 是否失败;如果没有问题就区分是在父亲还是儿子那里继续往下走逻辑链路。注意每次都要记得关闭不再使用的文件句柄以免泄漏资源! --- ### 总结 综上所述,掌握了 `fork()` 不仅可以加深对操作系统底层运作的理解还能帮助开发者设计更加灵活高效的软件架构方案。不过需要注意的是由于涉及到上下文切换开销较大因此不建议频繁调用除非确实有必要这样做才行哦!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值