fork函数详解

fork函数详解首先了解什么是 fork 一个进程 包括代码 数据和分配给进程的资源 fork 函数通过系统调用创建一个与原来进程几乎完全相同的进程 也就是两个进程可以做完全相同的事 但如果初始参数或者传入的变量不同 两个进程也可以做不同的事 一个进程调用 fork 函数后 系统先给新的进程分配资源 例如存储数据和代码的空间 然后把原来的进程的所有值都复制到新的新进程中 只有少数值与原来的进程的值不同 上面的话通俗理解就是 fork 是复制进程的函数 程序一开始就会产生一个进程 当这个进程 代码 执行到 fork

首先了解什么是fork?

d96e35c8a538ff7b0017047252b92768.png

上面的话通俗理解就是: fork是复制进程的函数,程序一开始就会产生一个进程,当这个进程(代码)执行到fork()时,fork就会复制一份原来的进程即就是创建一个新进程,我们称子进程,而原来的进程我们称为父进程,此时父子进程是共存的,他们一起向下执行代码。

注意的一点:就是调用fork函数之后,一定是两个进程同时执行fork函数之后的代码,而之前的代码以及由父进程执行完毕。

fork的特点:

首先明白linux中:

PID表示的进程号,是唯一的,一个PID只标识一个进程

PCB:进程控制块,进程控制块是用一个结构体struct task_struct来实现

31b49ff7cb964a5b940d5408fe15be37.png

fork的返回值问题:

在父进程中,fork返回新创建子进程的进程ID;

在子进程中,fork返回0;

如果出现错误,fork返回一个负值;  

getppid():得到一个进程的父进程的PID;

getpid():得到当前进程的PID;

*注意:在fork函数执行完毕后,如果创建新进程成功,则出现两个进程,一个是子进程,一个是父进程。在子进程中,fork函数返回0,在父进程中,fork返回新创建子进程的进程ID。我们可以通过fork返回的值来判断当前进程是子进程还是父进程。

fork是把已有的进程复制一份,当然把PCB也复制了一份,然后申请一个PID

子进程的PID=父进程的PID+1;

bf2ba936cfb649ab8c5e0606fabd3af1.png

下面我们举一个简单的例子:

第一次看的时候非常的奇怪,一个函数返回两次?是的,在调用fork后,fork函数后面的所有代码会执行两遍。

#include 
  
    #include 
   
     #include 
    
      / *最基础的fork例子 / int main(int argc, char const *argv[]) { pid_t pid; //判断1 if ((pid=fork()) < 0) { perror("fork error"); } //判断2 else if (pid == 0)//子进程 { printf("child getpid()=%d\n", getpid()); } //判断3 else if(pid > 0)//父进程 { printf("parent getpid()=%d\n", getpid()); } return 0; } 
     
    
  

结果如下:

parent getpid()=13725 child getpid()=13726 

*注意:这里的pid_t类似一个类型,就像int型一样,int型定义的变量都是整型的,pid_t定义的类型都是进程号类型。这个语句的意思是定义了一个pid_t类型的变量pid,fork()函数返回一个进程号,这个进程号赋给了pid。pid_t在头文件types.h(sys/types.h)中定义

pid_t就是一个short类型变量,实际表示的是内核中的进程表的索引

试试判断下面代码:

 #include 
  
    #include 
   
     #include 
    
      int main() { pid_t fpid;//fpid表示fork函数返回的值 int count=0; fpid=fork(); if(fpid<0) printf("error in fork!"); else if(fpid==0) { printf("我是子进程,id:%d\n",getpid()); count++; } else { printf("我是父进程,id:%d\n",getpid()); count++; } printf("统计结果是:%d\n",count); exit(0); } 
     
    
  

4daf60e5731e4d37bb893cee07fe533c.png

 父子进程的调用流程:

下面我们讲解一下fork调用的细节

int main(){ fork();//fork1 fork();//fork2 printf("love\n"); return 0; } 

上述代码打印了4次love,创建了4个进程(1一个父进程,3个子进程)

581bff9bef734e929209386343aaf074.png

假设我们的main进程pid是1001,注意看左边的1,2,4进程其实都是main进程1001。进程3,6是同一个进程1002。所有一共有1001,1002,1003,1004四个进程。也就是只要数叶子节点就行了。其中1个是main进程,其它3个是子进程。有多少个进程就输出多少次hello字符串。也就是只有4,5,6,7执行了printf。

如果明白了上面的过程下面我给出四个例子和解释,可以自己先试着判断

eg1:

int main() { int n=2; for(;i 
   

13a43ca4f51348fe8a67ed2d30f9ca0a.png

eg2:

int main() { int n=2; for(;i 
   

fork函数详解

eg3:

int main() { fork()||fork(); printf("A\n"); exit(0); 1)在父进程中,fork返回新创建子进程的进程ID;大于0的 2)在子进程中,fork返回0; 3)如果出现错误,fork返回一个负值; }

结果打印3个A,共创建3个进程

fork()给子进程返回一个零值,而给父进程返回一个非零值

在main这个主进程中,首先执行 fork() || fork(), 左边的fork()返回一个非零值,根据||的短路原则,前面的表达式为真时,后面的表达式不执行,故包含main的这个主进程创建了一个子进程,

由于子进程会复制父进程,而且子进程会根据其返回值继续执行,就是说,在子进程中, fork() ||fork()这条语句 左边表达式的返回值是0, 所以||右边的表达式要执行,这时在子进程中又创建了一个进程,

即main进程->子进程->子进程,一共创建了3个进程。

a491461cc6934bf3ad3d73a20a5c971e.png  

eg4:

int main() { fork()&&fork(); printf("A\n"); exit(0); }

结果输出3个A,创建3个进程

57c3ee87567647afa62ab471f2ad2624.png

注意小tips:

父子进程相同:

  • 刚刚fork后,data段,text段,堆,栈,环境变量,全局变量,宿主目录位置,进程工作目录,信号处理方式

父子进程不同:

  • 进程id,返回值,各自父进程,进程创建时间,闹钟,未决信号

父子进程共享:

  • 文件描述符
  • mmap映射区
  • 读时共享,写时复制—————–全局变量

对fork复制进程做了一个优化—-写时拷贝技术。写时拷贝指的是两个任务可以同时自由读取内存,但任意一个任务试图对内存进行修改时,内存就会复制一份提供给修改方单独使用,以免影响到其他的任务使用。

fork函数详解

fork函数详解

特别的,fork之后父进程先执行还是子进程先执行不确定,取决于内核所使用的调度算法。

注意父进程多次fork后不加以控制,我们会发现打印结果不唯一,无序。是因为对于操作系统将代码交给cpu执行的时候产生的子进程相当于同时产生的,并发运行,他们站在同一起跑线上去争夺cpu,谁抢到了谁就去运行打印数据,这是与系统调度有关,想让他们有序,可以加入sleep函数让其休眠一下

看到这里相信对与fork就有了基本的了解,点赞收藏是对博主最大的支持!有任何问题欢迎评论区留言讨论! 

f4fc38606e44ae26d8a9ebd2e258abf5.png

往期回顾:Linux常用命令及操作演示(细节满满)

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/200485.html原文链接:https://javaforall.net

(0)
上一篇 2026年3月20日 上午10:44
下一篇 2026年3月20日 上午10:44


相关推荐

  • Angular 面试题汇总2-Component/Service (Angular v8+)

    Angular 面试题汇总2-Component/Service (Angular v8+)Angularv8+面试系列Angular面试题汇总1-基本知识.目录关于AngularComponentcss样式的作用域、ShadowDOM关于AngularService单例服务(singleton)forRoot()模式关于AngularComponentcss样式的作用域、ShadowDOMShadowDOM是HTML规范的一部分,它允许开发人员封装自己的HTML标记,CSS样式和JavaScript。创建样式Component时,可以通过设置,启用。@Com.

    2022年10月17日
    3
  • 如何在pycharm中安装第三方库_pycharm使用pip安装pygame

    如何在pycharm中安装第三方库_pycharm使用pip安装pygame一、场景描述:在开发python小游戏前,需要安装pygame的插件,如何在pycharm软件中安装pygame的插件呢,本文将重点讲解如何安装pygame这个插件。二、安装步骤:1、在pycharm软件中定位到file–settings2、定位到:project(自己的项目中)–pythoninterpreter,选择右边的pip3、双击pip,进入查询插件界面,输入pygame,进行查询这个插件,最后点击installpackage4、安装成功后,会在pycharm软件的下方提示安

    2022年8月26日
    8
  • String字符串截取几种方法

    String字符串截取几种方法String截取字符串publicstaticvoidmain(String[]args){ Stringa="abcd-efg"; Stringa1=a.substring(a.lastIndexOf("-")+1); Stringa2=a.substring(0,a.indexOf("-")); System.out.println(a1);//efg …

    2022年5月19日
    53
  • 2017中国程序员薪资生存现状调查报告结论_程序员的收入

    2017中国程序员薪资生存现状调查报告结论_程序员的收入程序员一直都是一个备受人们关注的群体。2014年,据IDC统计,全球约有1850万名程序员,中国占10%。随着近年全国互联网创业热潮的兴起,“互联网+”、“云计算”以及“智能硬件”等领域发展迅速,市场对程序员的需求更为旺盛。  由程序员客栈联合稀土掘金通过对北京、广东、浙江、上海等全国28个省、直辖市及特别行政区的10W+优秀程开发者进行了一次调查。调查报告里对程序员的年龄组成、性别比例、

    2022年10月11日
    3
  • AI智能体扣子(Coze)工作流搭建,试卷出题系统,涵盖所有学科、所有年级,保姆级教程

    AI智能体扣子(Coze)工作流搭建,试卷出题系统,涵盖所有学科、所有年级,保姆级教程

    2026年3月13日
    2
  • linux查看ssh信息,linux查看与开启ssh「建议收藏」

    首先通过物理终端进入到linux上,手工检查ssh发现没运行/etc/init.d/sshdstatussshdisstopped手工启动服务,发现报告权限错误。/etc/init.d/sshdstartStartingsshd:/var/empty/sshdmustbeownedbyrootandnotgrouporworld-writable.[FAILED]使用…

    2022年4月14日
    67

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号