fork函数详解_fork()(1)

收集整理了一份《2024年最新物联网嵌入式全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升的朋友。
img
img

如果你需要这些资料,可以戳这里获取

需要这些体系化资料的朋友,可以加我V获取:vip1024c (备注嵌入式)

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人

都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

}

return 0;

}


结果如下:



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


两个判断的代码都执行了,这是非常不可思议的,但fork函数确实实现了这样的功能。也就是在fork函数后面的代码都会执行2遍。 这就是为什么两个判断都会被执行的原因。  
 现在来梳理一下成功fork的执行流程  
 第一步: pid=fork(),如果成功那么pid就有一个非0正值。否则返回-1。  
 第二步: 因为pid>0,所以进入判断3。这是在父进程。  
 第三步: 父进程的代码执行完了,程序又会把fork后面的函数再执行一遍,此时pid的值变为0,所以进入判断2。


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


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


**试试判断下面代码:**



#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>

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](https://img-blog.csdnimg.cn/4daf60e5731e4d37bb893cee07fe533c.png)


####  **父子进程的调用流程:**


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



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


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


![581bff9bef734e929209386343aaf074.png](https://img-blog.csdnimg.cn/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<n;i++)
>     {
>         fork();
>         printf("A\n");//遇到\n会自动刷新缓冲区
>     }
>     exit(0);
> }
> ```
> 
> ![13a43ca4f51348fe8a67ed2d30f9ca0a.png](https://img-blog.csdnimg.cn/13a43ca4f51348fe8a67ed2d30f9ca0a.png)
> 
> 
> 
> eg2:
> 
> 
> 
> ```
> int main()
> {
>     int n=2;
>     for(;i<n;i++)
>     {
>         fork();
>         printf("-");//不会刷新缓冲区
>     }
>     exit(0);
> }
> ```
> 
> ![](https://img-blog.csdnimg.cn/34aedb9eee3241ab99e53dd4084a1aca.png)
> 
> 
> 
> 
> 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](https://img-blog.csdnimg.cn/a491461cc6934bf3ad3d73a20a5c971e.png)  
> 
> 
> eg4:
> 
> 
> 
> ```
> int main()
> {
> 	fork()&&fork();
> 	printf("A\n");
> 	exit(0);
> }
> ```
> 
> **结果输出3个A,创建3个进程**
> 
> 
> ![57c3ee87567647afa62ab471f2ad2624.png](https://img-blog.csdnimg.cn/57c3ee87567647afa62ab471f2ad2624.png)
> 
> 
> 
> 



> 
> ### **注意小tips:**
> 
> 
> 父子进程相同:
> 
> 
> * 刚刚fork后,data段,text段,堆,栈,环境变量,全局变量,宿主目录位置,进程工作目录,信号处理方式
> 
> 
> 父子进程不同:
> 
> 
> * 进程id,返回值,各自父进程,进程创建时间,闹钟,未决信号
> 
> 
> 父子进程共享:
> 
> 
> * 文件描述符


**收集整理了一份《2024年最新物联网嵌入式全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升的朋友。**
![img](https://img-blog.csdnimg.cn/img_convert/3787bd3da456e3b11eed6af6a3a84a61.png)
![img](https://img-blog.csdnimg.cn/img_convert/32e9ee51c4501b612cf7ab66c906662a.png)

**[如果你需要这些资料,可以戳这里获取](https://bbs.csdn.net/topics/618679757)**

**需要这些体系化资料的朋友,可以加我V获取:vip1024c (备注嵌入式)**

**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人**

**都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**

v0j-1715873043448)]

**[如果你需要这些资料,可以戳这里获取](https://bbs.csdn.net/topics/618679757)**

**需要这些体系化资料的朋友,可以加我V获取:vip1024c (备注嵌入式)**

**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人**

**都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**

  • 5
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值