Linux——进程(2)——fork


一、fork创建进程

使用fork函数创建一个进程
pid_t fork(void);

fork函数调用成功,返回两次
返回值为0, 代表当前进程是子进程
返回值非负数, 代表当前进程为父进程

调用失败,返回-1

#include<stdio.h>
#include<sys/types.h>
#include<unistd.h>
int main()
{
        pid_t pid;
		
        printf("father id:%d\n",getpid());
		//在没有调用fork函数之前先看一下进程号是多少
        pid=fork();
		//用pid接收fork的返回值
        if(pid>0){
                printf("this is father print,pid = %d\n",getpid());
        }//做个判断验证一下fork函数创建的进程他们的返回值
        else if(pid==0){
                printf("this is child print,child pid is:%d\n",getpid());
        }

        return 0;
}

在这里插入图片描述

二、fork的返回值 以及 父子进程的数据交互方式

这里利用代码可以了解 父进程的返回值retpid正好是子进程的ID号

#include<stdio.h>
#include<sys/types.h>
#include<unistd.h>
int main()
{
        pid_t pid;
        pid_t pid2;
        pid_t retpid;
        int data=10;

        pid = getpid();
        printf("before pid is:%d\n",pid);

        retpid=fork();
		//定义一个变量接收fork的返回值
        pid2=getpid();
        printf("after pid is:%d\n",pid2);

        if(pid==pid2){
                printf("this is father print,retpid is:%d\n",retpid);
        }else{
                printf("this is child print,child pid is:%d,retpid is:%d\n",getpid(),retpid);
                data+=10;
        }
        printf("data is:%d\n",data);
        return 0;
}

在这里插入图片描述
同时 在这里我们利用data的值还证明了一个问题 在旧的Linux系统中的copy方式为 全copy 新的 则为copy on write 如果子进程对值有做修改 则copy相应的值修改 如果没有则为共享数据原则

三、fork创建子进程的一般目的

(1)一个父进程希望复制自己,使父、子进程同时执行不同的代码段。
这在网络服务进程中是常见的——父进程等待客服端的服务请求。当这种请求到达时,父进程调用fork,使子进程处理此请求。父进程则继续等待下一个服务请求到达。

这里做个代码模拟一下场景

#include<stdio.h>
#include<sys/types.h>
#include<unistd.h>
int main()
{
        pid_t pid;
        int data;

        while(1){
                printf("please enter a data:\n");
                scanf("%d",&data);
                if(data==1){//当输入为1时创建进程
                        pid=fork();
                        if(pid>0){
									//父进程闲置
                        }
                        else if(pid==0){
                                while(1){
                                        printf("do net requst,pid=%d\n",getpid());
                                        //子进程打印自己的进程id号
                                        sleep(3);  //间隔3秒
                                }
                        }
                }
                else{
                        printf("wait,do nothing\n");  //输入其他数字 反馈信息
                }
        }
        return 0;
}

在这里插入图片描述
(2)一个进程要执行一个不同的程序。这对shell是常见的情况。在这种情况下,子进程从fork返回后立即调用exec。

总结

由fork创建的新进程被称为子进程(child process)。fork函数被调用一次,但返回两次。两次返回的唯一区别是子进程的返回值是0,而父进程的返回值是新子进程的进程ID。
子进程和父进程继续执行fork调用之后的指令。子进程是父进程的副本。例如,子进程获得父进程数据空间、堆和栈的副本。
由于在fork之后经常跟随着exec,所以现在的很多实现并不执行一个父进程数据段、栈和堆的完全复制。作为替代,使用了写时复制(Copy-On-Write,COW)技术。这些区域由父、子进程共享,而且内核将它们的访问权限改变为只读的。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值