Linux进程【3】fork函数与进程等待(超详解哦)

fork与进程等待

  • 引言
  • fork
    • fork创建子进程的过程
    • 写时拷贝
  • 进程等待
    • wait
    • waitpid
    • 阻塞等待与非阻塞轮询
  • 总结

引言

fork函数在Linux中是一个非常重要的系统调用接口!它用于在当前的已有进程中创建一个新的进程(子进程)。再由父子进程并发地执行不同地代码块,就相当于父子进程给子进程派了一块代码让他去执行。
在子进程执行完代码块后,应该给父进程一个发聩,这个时候就需要父进程去等待子进程,然后回收子进程,以免形成内存泄漏等问题。
接下来就来详细地介绍fork函数以及进程等待:

fork

fork可以从当前进程中创建一个新进程,已有的进程就是父进程,新进程就是子进程,父进程与子进程并发地执行不同的代码块:

#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>

int main()
{
    pid_t rid = 0;

    rid = fork();
    if(rid < 0)
    {
        perror("fork:");
    }
    else if(rid == 0) //子进程
    {
        printf("i am child\n");
    }
    else //父进程(rid > 0)
    {
        printf("i am parent\n");
    }
    return 0;
}

在这里插入图片描述

fork创建子进程的过程

在创建子进程时:
操作系统会给子进程分配新的内存块与内核数据结构;
然后父进程的部分数据被拷贝到子进程;
然后子进程会被操作系统添加到调度列表中;
最后会分别返回值给父子进程,对父进程返回子进程的pid,对子进程返回0

#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>

int main()
{
    pid_t rid = 0;

    int a = 10;
    printf("before: %d\n", a); 

    rid = fork(); //创建子进程
    ++a;
    printf("rid: %d: after: %d\n", rid, a);

    if(rid == 0) //子进程
    {
        ++a;
        printf("child: %d\n", a);
    }
    else if(rid > 0) //父进程
    {
        a+=2;
        printf("parent: %d\n", a);
    }
    return 0;
}

在这里插入图片描述

在这段代码中,定义了一个变量a,我们可以通过这个变量a的变化来验证fork创建新进程的过程:

首先打印了一遍before,此时a的值为10。说明这时只有父进程一个执行流在执行代码;
然后发现after打印了两遍,两遍a的值都是11,由父子进程分别打印。这首先说明fork之后有父子进程两个执行流在执行代码。并且在创建子进程时,子进程获取到了父进程之前的变量a,所以两个执行流在这里打印出的值都是11;
然后if_else对代码进行了分流,父进程打印a+=2后的结果13,子进程打印a++后的结果12。说明进程之间是独立的,他们有自己的进程地址空间与页表,转化到不同的物理内存,对自己进程中数据的改变不会影响对方进程。
在这里插入图片描述
需要注意的是,父子进程的调度先后,完全由调度器决定

写时拷贝

通过上面的介绍,我们知道在创建子进程时,父进程要将自己的数据拷贝给子进程。但是对于代码或者子进程没有进行修改的数据,在物理内存中在将这些数据存储一份显然是浪费内存空间的。

所以父子进程在拷贝数据是是以写时拷贝的方式来进行的
创建子进程时,父进程将自己的进程地址空间与页表拷贝给子进程,即父子进程的数据段和代码段在通过页表映射后指向同一块物理内存:
在这里插入图片描述

如果子进程中会对一些数据做修改时,就会发生写时拷贝。即将要修改的数据拷贝一份到另外的物理空间,页表的转化关系也指向这块新的物理空间,再由子进程对其进行修改:
在这里插入图片描述

通过这样的写时拷贝的方式,就可以减少内存的浪费。

进程等待

子进程退出后,父进程应该获取子进程的退出状态,看看子进程是否正常退出,如果出现异常的错误码是什么;
父进程也应该回收子进程的资源,如果子进程的资源没有被回收,就会造成内存泄漏;
没有被父进程回收的子进程就会成为 “僵尸进程”,无法被杀死。

父进程可以通过waitwaitpid函数来对子进程进行等待:
在这里插入图片描述

wait

pid_t wait(int* status) 用于等待任一子进程
等待成功返回子进程pid,等待中出错返回-1,errno被设置;
参数为输出型参数,用于获取子进程的退出状态,由操作系统填充(若不关心返回状态,参数设为NULL即可)。

在等待结束后,可以通过 WIFEXITED(status);查看进程是否是正常退出,若为正常终止子进程,则为真;
通过 WEXITSTATUS(status);查看进程的退出码,若WIFEXITED返回真,提取子进程退出码:

#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>

int main()
{
    pid_t rid = 0;

    rid = fork();
    if(rid == 0) //子进程
    {
        printf("i am child\n");
        sleep(5);
    }
    else if(rid > 0) //父进程
    {
        printf("i am parent\n");

        int status = 0;
        pid_t ret = 0;
        ret = wait(&status); //等待

        if(WIFEXITED(status) != 0 && ret == rid) //如果等待成功打印子进程退出码
        {
            printf("child return :%d\n", WEXITSTATUS(status));
        }
        else
        {
            return 1;
        }
    }
    return 0;
}

在这里插入图片描述

waitpid

pid_t waitpid(pid_t pid, int *status, int options); 可以等待任一子进程,也可以用于等待指定子进程。等待成功返回子进程pid,等待中出错返回-1,errno被设置;

第一个参数pid表示指定要等待子进程的pid,若要等待任一子进程,则传参-1;
第二个参数为输出型参数,参数为输出型参数,用于获取子进程的退出状态,由操作系统填充(若不关心返回状态,参数设为NULL即可);
第三个参数为等待状态,有多种选项:当设置为WNOHANG:,表示若pid指定的子进程没有结束,则waitpid函数返回0,不予以等待(即非阻塞等待)。

#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>

int main()
{
    pid_t rid = 0;

    rid = fork();
    if(rid == 0)
    {
        printf("i am child\n");
        sleep(5);
    }
    else if(rid > 0)
    {
        printf("i am parent\n");

        int status = 0;
        pid_t ret = 0;
        ret = waitpid(rid, &status, 0); //阻塞式等待pid为rid的子进程

        if(WIFEXITED(status) != 0 && ret == rid)
        {
            printf("child return :%d\n", WEXITSTATUS(status));
        }
        else
        {
            return 1;
        }
    }
    return 0;
}

在这里插入图片描述

阻塞等待与非阻塞轮询

在使用waitwaitpid进行进程等待时,若子进程正在执行,父进程就会阻塞式的等待子进程执行结束,等待成功后再继续执行接下来的代码:

int main()
{
    pid_t rid = 0;

    rid = fork();
    if(rid == 0)
    {
        int n = 10;
        while(n--)
        {
            printf("i am child, %d\n", n);
            sleep(1);
        }
    }
    else if(rid > 0)
    {
        int status = 0;
        pid_t ret = 0;
        ret = waitpid(rid, &status, 0); //阻塞式等待,成功等待子进程后才会执行后面的代码
                
        if(WIFEXITED(status) != 0 && ret == rid)
        {
            printf("child return :%d\n", WEXITSTATUS(status)); 
        }
        else
        {
            return 1;
        }
    }
    return 0;
}

在这里插入图片描述

但是如果父进程阻塞等待子进程的时间过长,就会影响代码的效率。
如果父进程在等子进程时发现子进程正在执行,父进程可以选择不阻塞等待,而是去执行别的代码,隔一段时间去看看子进程有没有退出。这样的非阻塞轮询的方式可以提高代码的效率:

非阻塞等待的方式可以通过waitpid函数的WNOHANG选项来实现:

int main()
{
    pid_t rid = 0;

    rid = fork();
    if(rid == 0)
    {
        int n = 10;
        while(n--)
        {
            printf("i am child, %d\n", n);
            sleep(1);
        }
    }
    else if(rid > 0)
    {
        int status = 0;
        pid_t ret = 0;
        
        do //非阻塞轮询,当成功等待时终止循环
        {
            ret = waitpid(rid, &status, WNOHANG);
            if(ret == 0)
            {
                printf("child is running, i do something...\n");
                sleep(1);
            }
        }while(ret == 0); 

        if(WIFEXITED(status) != 0 && ret == rid)
        {
            printf("child return :%d\n", WEXITSTATUS(status));
        }
        else
        {
            return 1;
        }
    }
    return 0;
}

在这里插入图片描述

这样就实现了父进程在等待子进程期间也可以做一些事,提高了效率(这里的父子进程是并发进行的,两个进程抢占显示器打印的先后是由调度器决定的,所以不是很按顺序属于正常)。

总结

到此,关于fork函数与进程等待的知识就介绍完了

如果大家认为我对某一部分没有介绍清楚或者某一部分出了问题,欢迎大家在评论区提出

如果本文对你有帮助,希望一键三连哦

希望与大家共同进步哦

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/330768.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

nodejs下载安装

一、node下载安装 官网下载 官网 根据自己电脑系统选择合适的版本进行下载&#xff0c;我这里选择window 64 位 下载完点击安装 打开cmd查看安装 此处说明下&#xff1a;新版的Node.js已自带npm&#xff0c;安装Node.js时会一起安装&#xff0c;npm的作用就是对Node.js…

2024年外贸新兴市场有哪些 | 箱讯科技国际贸易平台

当前欧美市场经济增速放缓&#xff0c;通胀持续高位导致物价普遍上涨&#xff0c;进一步引发消费疲软。此外&#xff0c;受原材料价格、劳动力、土地等经营成本上升影响&#xff0c;外贸出口企业利润被进一步压缩。 困顿之中&#xff0c;新兴市场成为破局关键&#xff0c;巨大的…

基于Python flask的猫眼电影票房数据分析可视化系统,可以定制可视化

技术方案 猫眼电影票房数据分析可视化系统是基于Python Flask框架开发的一款用于分析和展示猫眼电影票房数据的Web应用程序。该系统利用Flask提供了一个简单而强大的后端框架&#xff0c;结合Request库进行网络爬虫获取猫眼电影票房数据&#xff0c;并使用Pyecharts进行可视化…

【前后端的那些事】评论功能实现

文章目录 聊天模块1. 数据库表2. 后端初始化2.1 controller2.2 service2.3 dao2.4 mapper 3. 前端初始化3.1 路由创建3.2 目录创建3.3 tailwindCSS安装 4. tailwindUI5. 前端代码编写 前言&#xff1a;最近写项目&#xff0c;发现了一些很有意思的功能&#xff0c;想写文章&…

Elasticsearch8 集群搭建(二)配置篇:(1)节点和集群配置

安装完Elasticsearch后&#xff0c;需要对其进行配置&#xff0c;包括以下几部分&#xff1a;节点和集群配置、系统配置、安全配置。 此篇记录节点和集群配置的内容&#xff0c;后续将更新系统配置和安全配置。 节点和集群配置&#xff1a; 通过编辑/usr/local/elasticsearc…

Midjourney Prompt 常用参数列表

完整参数列表 参数名称调用方法使用案例注意事项V5V4V3niji版本在关键词后加空格&#xff0c;然后带上版本参数&#xff1a; --v 或者 —v--version 或者 —versionvibrant california poppies --v 5版本仅支持 1、2、3、4、5。长宽比在关键词后加空格&#xff0c;然后带上长…

大寒吃什么食物养生?养生食物清单记在手机便签更方便

随着冬季的深入&#xff0c;我们迎来了二十四节气中的最后一个——大寒。寒风凛冽&#xff0c;白雪皑皑&#xff0c;这是大自然在提醒我们&#xff0c;要更加注重身体的保养。而大寒时节&#xff0c;选择对的食物&#xff0c;就显得尤为重要。 每到大寒&#xff0c;我都会精心…

L1-035 情人节(Java)

以上是朋友圈中一奇葩贴&#xff1a;“2月14情人节了&#xff0c;我决定造福大家。第2个赞和第14个赞的&#xff0c;我介绍你俩认识…………咱三吃饭…你俩请…”。现给出此贴下点赞的朋友名单&#xff0c;请你找出那两位要请客的倒霉蛋。 输入格式&#xff1a; 输入按照点赞…

C++代码入门02:Vector中的push_back

图源&#xff1a;文心一言 上机题目练习整理&#xff0c;本篇作为线性表的代码补充&#xff0c;提供了两种&#xff08;差别并不大&#xff09;算法&#xff0c;供小伙伴们参考~&#x1f95d;&#x1f95d; 第1版&#xff1a;在力扣新手村刷题的记录 方法一&#xff1a;自己写…

Android Studi安卓读写NDEF智能海报源码

本示例使用的发卡器&#xff1a;https://item.taobao.com/item.htm?id615391857885&spma1z10.5-c.w4002-21818769070.11.1f60789ey1EsPH <?xml version"1.0" encoding"utf-8"?> <androidx.constraintlayout.widget.ConstraintLayout xmln…

vscode调试debug,launch.json文件‘args’无法发传递给脚本

问题&#xff1a;调试时&#xff0c;脚本执行&#xff0c;发现在launch.json文件中明明定义了“args”参数&#xff0c;却没有传递给执行命令。 解决&#xff1a; launch.json中的"name"参数不要随便起&#xff0c;要与执行的文件名一致&#xff01; 参考链接&…

品牌全球化:关于跨界合作的探索与解析

在全球化的时代背景下&#xff0c;品牌出海已经成为企业发展的重要战略之一。然而&#xff0c;面对文化差异、市场竞争和消费者需求等多重挑战&#xff0c;品牌如何成功地打入海外市场&#xff0c;是许多企业面临的难题。跨界合作作为一种新兴的商业模式&#xff0c;正逐渐成为…

旅游平台day02

1. 用户注册 概述&#xff1a; 常见的注册方式&#xff1a;邮箱注册、手机号注册、昵称注册、或者以上几种同时支持 本项目仅仅支持手机号注册 需求&#xff1a; 项目启动后&#xff0c;访问regist.html进入注册页面 手机号校验 前后台都需要对手机号进行校验 前端校验&am…

微信小程序+前后端开发学习材料2-(视图+基本内容+表单组件)

学习来源 视图 1.swiper 滑块视图容器。其中只可放置swiper-item组件&#xff0c;否则会导致未定义的行为。 显示面板指示点indicator-dots 基础内容 1.icon 图标组件 实例演示 2.progress 进度条。组件属性的长度单位默认为px&#xff0c;咱用rpx。 实例演示 这…

selenium爬虫爬取当当网书籍信息 | 最新!

如果对selenium不了解的话可以到下面的链接中看基础内容&#xff1a; selenium爬取有道翻译-CSDN博客 废话不多说了下面是代码并且带有详细的注释&#xff1a; 爬取其他类型的书籍和下面基本上是类似的可以自行更改。 # 导入所需的库 from selenium import webdriver from …

node.js(express.js)+mysql实现注册功能

文章目录 实现步骤一、获取客户端提交到服务器的用户信息&#xff0c;对表单中的数据&#xff0c;进行合法性的效验 代码如下:二、检测用户名是否被占用三、对密码进行加密四、插入新用户&#xff08;完整代码&#xff09;总结 实现步骤 一、获取客户端提交到服务器的用户信息…

Vue3中动态组件使用

一&#xff0c;动态组件使用&#xff1a; 应用场景&#xff1a;动态绑定或切换组件 应用Vue3碎片&#xff1a; is 1.使用 a.组件A <div class"layout-base"><Button>红茶</Button> </div>a.组件B <div class"layout-base"&g…

【深度强化学习】目前落地的挑战与前沿对策

到目前为止&#xff0c;深度强化学习最成功、最有名的应用仍然是 Atari 游戏、围棋游戏等。即使深度强化学习有很多现实中的应用&#xff0c;但其中成功的应用并不多。为什么呢&#xff1f;本文总结目前的挑战。 目录 所需的样本数量太大探索阶段代价太大超参数的影响非常大稳定…

【MATLAB源码-第115期】基于matlab的QSM正交空间调制系统仿真,输出误码率曲线。

操作环境&#xff1a; MATLAB 2022a 1、算法描述 正交空间调制&#xff08;QSM&#xff09;是一种先进的无线通信技术&#xff0c;它通过利用发射端的多天线阵列来传输信息&#xff0c;从而提高了数据传输的效率和速率。这种技术的关键在于它使用天线阵列的空间特性来编码额…