【Linux】fork()

本文介绍了Linux下fork()函数的基本概念及使用方法,并通过一个简单示例程序展示了如何使用fork()来创建子进程,同时解释了父子进程间变量独立性的原理。

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

fork()是LinuxC中一个基本函数,他存在于头文件<unistd.h>之中,用于创造一个子进程。说白了,就是将本程序一分为二,互补影响地执行。

直接贴一段简单的程序说明fork()的使用:

#include <unistd.h>  
#include <stdio.h>   
int main(){
    int count=0;
    pid_t fpid=fork();
    if(fpid<0){   
        printf("创建父子进程失败!");
    }
    else if(fpid==0){
        printf("子进程ID:%d\n",getpid());
        count++;
    }  
    else{
        printf("父进程ID:%d\n",getpid());
        count++;
    }  
    printf("count=%d\n",count);
    waitpid(fpid,NULL,0);
    return 0;
}

这段程序的运行结果如下图所示:


整个程序是这样的,在一开始定义了一个count变量。

利用fork()函数将整个程序分成了两半,在pid_t fpid==0是子进程执行的分支,而另一半则是父进程执行的分支。int count=0这个变量被原封不动地拷贝到这两个分支之中。

互补影响地做自加。

之后,这两个分支同时执行printf("count=%d\n",count);这段代码打印count。最后用waitpid(fpid,NULL,0);结束fork(),否则这个程序不知道在Ubuntu里面为何无法自然而言地结束。

用一个简单的流程图可以完美地说明这个问题:


一个进程调用fork()函数后,系统先给新的进程分配资源,例如存储数据和代码的空间。然后把原来的进程的所有值都复制到新的新进程中,只有少数值与原来的进程的值不同。相当于克隆了一个自己。 

那么能否经历过fork()的父子进程同时操作一个变量。就像上述的程序,将count弄到2呢?比如将int count=0弄成全局变量的。答案是不行的。fork后子进程中是父进程的完全复制(其中有写时复制技术),不管什么变量,fork后父子进程中都是一样的,但两者之间没有关系,任何一个进程修改变量后,在另一个进程中都不能知道,更不能访问另一个进程中的变量,即使是全局变量。

如果你需要两个进程(线程)共同操作一个变量,实现一种互斥的关系,请用pthread_create(),具体见《【Linux】线程互斥》(点击打开链接),fork()更多的是,配合管道在描述一种线程之间的同步关系。

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

余额充值