当前位置: 首页 > news >正文

【Linux系统化学习】进程等待

目录

进程等待

进程等待的必要性

进程等待的方法

wait方法

等待一个进程(阻塞等待) 

waitpid方法

任意等待多个进程(阻塞等待)

父进程获取子进程的退出信息

非阻塞轮询等待


进程等待

进程等待的必要性

  • 之前讲过,子进程退出,父进程如果不管不顾,就可能造成‘僵尸进程’的问题,进而造成内存泄漏
  • 另外,进程一旦变成僵尸状态,那就刀枪不入,“杀人不眨眼”的kill -9 也无能为力,因为谁也没有办法杀死一个已经死去的进程
  • 最后,父进程派给子进程的任务完成的如何,我们需要知道。如,子进程运行完成,结果对还是不对,或者是否正常退出。
  • 父进程通过进程等待的方式,回收子进程资源,获取子进程退出信息

总结:进程等待就是父进程解决僵尸进程的


进程等待的方法

父进程通过wait/waitpid系统调用来实现

wait方法

#include<sys/types.h>
#include<sys/wait.h>pid_t wait(int*status);
  • 返回值:成功返回被等待进程pid,失败返回-1。
  • 参数:输出型参数,获取子进程退出状态,不关心则可以设置成为NULL

等待一个进程(阻塞等待) 

#include<stdio.h>2 #include<unistd.h>3 #include<sys/types.h>4 #include<sys/wait.h>5 #include<stdlib.h>6 void Worker()7 {8     int cut =5 ;9     while(cut)10     {11          printf(" am a child process; p    id : %d ppid : %d ,cut : %d \n",getpid(    ),getppid(),cut--);12          sleep(1);13     }14 }15 int main()16 {17     pid_t id = fork();18     if(id==0)19     {20         //child21         Worker();22         exit(0);23     }24     else{25         //father26         sleep(7);27         printf("wait before\n");28        pid_t rid = wait(NULL);29        printf("wait after\n");30         if(rid == id )31         {32             printf("wait success , pid     %d rpid:%d\n",getpid(),rid);33         }34         sleep(2);                      35     }36     return 0;37 }

总结:

  • 父进程通过wait子进程可以回收子进程的僵尸状态
  • 如果子进程没有退出,父进程必须进行阻塞等待(等待软件资源就绪),直接到子进程僵尸,wait回收,返回。
  • 父子进程水仙运行我们不直到是由于调度器决定的,但是我们可以很清楚的直到是父进程最后退出的。

waitpid方法

#include <sys/types.h>
#include <sys/wait.h>pid_t waitpid(pid_t pid, int* status, int options);

返回值

  • 当正常返回的时候waitpid返回收集到的子进程的进程ID;
  • 如果设置了选项WNOHANG,而调用中waitpid发现没有已退出的子进程可收集,则返回0;
  • 如果调用中出错,则返回-1,这时errno会被设置成相应的值以指示错误所在;

参数

pid

  • pid=-1,等待任一个子进程。与wait等效。
  • pid>0.等待其进程ID与pid相等的子进程。

status

  • WIFEXITED(status): 若为正常终止子进程返回的状态,则为真。(查看进程是否是正常退出)
  • WEXITSTATUS(status): 若WIFEXITED非零,提取子进程退出码。(查看进程的退出码)

options:

  • 0:表示父进程以阻塞的方式等待子进程,即子进程如果处在其它状态,不处在僵尸状态(Z状态),当子进程运行结束,操作系统会检测到,把父进程重新唤醒,然后回收子进程;
  • WNOHANG:非阻塞轮询等待,若 pid 指定的子进程没有结束,处于其它状态,则 waitpid() 函数返回0,不予等待。若正常结束,则返回该子进程的 ID;

任意等待多个进程(阻塞等待)

#include <sys/types.h>    5 #include <sys/wait.h>    6     7 #define N 5     8 void Worker(int i)    9 {    10     int cnt = 5;    11     while(cnt--)    12     {    13         printf("I am child, pid:%d, ppid:%d, %d\n", getpid(), get    ppid(), i);    14         sleep(1);    15     }    16     return;    17 }    18 int main()    19 {    20     int i=0;21     for( i = 0; i < N; i++)    22     {    23         //创建一批子进程24         pid_t id = fork();    25         if(id == 0)26         {27             // child     28             Worker(i);29             exit(0);30         }                                                          31         // father32 33     }34     //等待子进程35     int j=0;36     for(j=0;j<N;j++)37     {38         pid_t rid = waitpid(-1,NULL,0);39         if(rid>0)40         {41             printf("wait %d success\n",rid);42         }43     }

父进程获取子进程的退出信息

在上篇文章进程的终止我们提到,创建进程肯定是完成某一项任务的;任务的结果就三种情况:

  • 代码运行完毕,结果正确
  • 代码运行完毕,结果不正确
  • 代码出异常了,被操作系统发送信号终止

因此,作为父进程通过等待获取这个退出信息;这个信息就是在传入的参数status中被写入。

我们可以看到这个参数的类型是一个整形指针,需要我们在父进程执行的代码块中定义一个整形变量,将这个整型变量取地址作为wait/waitpid的参数交给操作系统,操作系统会将子进程的退出码填充给这个参数。 

  • wait 和 waitpid 都有一个 status 参数,该参数是一个输出型参数,由操作系统填充。

  • 如果传递 NULL,表示不关心子进程的退出状态信息。

  • 否则,操作系统会根据该参数,将子进程的退出信息反馈给父进程。

  • status 不能简单的当做整形来看待,可以当做位图来看待,具体细节如下图(只需要关注 status 低16比特位)

 

退出的情况可以总结为两种,正常终止(0:代表成功;非0:代表失败),异常终止; 

前7位代表异常终止时的终止信号,后8位代表我们正常终止的退出状态;

操作系统没有0号信号,因此,如果低七位是0说明子进程没有收到任何信号。

我们要想拿到这两个数字就要进行位运算

  • 退出码:exit_code = (status>>8)&0xFF
  • 信号码: sig_code= (status)& 0x7F

这样的方式非常的麻烦;我们可以使用系统中提供的转化方法获取退出码。

  • WIFEXITED(status): 若为正常终止子进程返回的状态,则为真。(查看进程是否是正常退出)
  • WEXITSTATUS(status): 若WIFEXITED非零,提取子进程退出码。(查看进程的退出码) 
#include<stdio.h>2 #include<unistd.h>3 #include<stdlib.h>4 #include<sys/types.h>5 #include<sys/wait.h>6 void Worker()7 {8     int cnt = 5;9     while(cnt)10     {11         printf("I am child process,pid: %d,ppid : %d\n,cnt: %d",getpid(),getppid(),cnt--    );12     }13 }14 int main()15 {16     pid_t id = fork();17     if(id == 0)18     {19         //child20         Worker();21        // exit(0);22        exit(10);23     }24     else{25         sleep(10);          26         //father30         int status = 0;31         printf("wait before\n");32        // pid_t rid = wait(NULL );33        pid_t rid = waitpid(id , &status ,0);34 35         printf("wait after\n");36         if(rid==id)37         {38             printf("wait success,pid: %d",getpid());                                                                                                                                    39         }40        // printf("%d\n",status);41        if(WIFEXITED(status))42        {43            printf("child process normal quit , exit code : %d \n",WEXITSTATUS(status));
44        }45        else46        {47            printf("child process quit except!!!\n");48        }49     }50     return 0;51 }

 


非阻塞轮询等待

前面说过如果子进程没有进入僵尸状态,父进程什么也不做就一直阻塞等待子进程。采用非阻塞的方式轮询等待的方式,子进程在没有进程僵尸状态的区间父进程可以进行其他的事情。

 #include<stdio.h>2 #include<stdlib.h>3 #include<unistd.h>4 #include<sys/types.h>5 #include<sys/wait.h>6 void Worker()7 {8     int cnt =3 ;9     while(cnt--)10     {11         printf("am a child  process pid : %d ppid %d num %d \n",getpid(),getppid(),cnt);12         sleep(2);13     }14 }15 int main()16 {17     pid_t id = fork();18     if(id==0)19     {20         //child21         Worker();22         exit(0);23     }24     else25     {26         //father27         while(1)28         {29             pid_t rid = waitpid(id,NULL,WNOHANG );30             if(rid == 0)31             {32                 printf("wait success,but process no quit\n");33                 printf("father process , to do athor thing.......\n");34                 sleep(1);                                                                                                                                                               35             }36             else if(rid < 0)37             {38                 printf("wait fail\n");39                 break;40             }41             else42             {43                 printf("wait success\n");44                 break;45             }46         }47     }48 49     return 0;50 }

第三个参数设置为WNOHONG后,可以通过循环语句控制父进程轮询等待;在子进程未就绪期间父进程可以执行其他的一些简单代码。


 今天对Linux下进程的等待的分享到这就结束了,希望大家读完后有很大的收获,也可以在评论区点评文章中的内容和分享自己的看法;个人主页还有很多精彩的内容。您三连的支持就是我前进的动力,感谢大家的支持!!! 

相关文章:

【Linux系统化学习】进程等待

目录 进程等待 进程等待的必要性 进程等待的方法 wait方法 等待一个进程(阻塞等待&#xff09; waitpid方法 任意等待多个进程&#xff08;阻塞等待&#xff09; 父进程获取子进程的退出信息 非阻塞轮询等待 进程等待 进程等待的必要性 之前讲过&#xff0c;子进程退…...

前端学习笔记 | HTML5+CSS3静态网页制作的技巧(持续更新)

注&#xff1a;本文的css样式均使用less写法 1、字体居中 &#xff08;1&#xff09;先text-align:center;行内元素水平居中 &#xff08;2&#xff09;再line-heigh:(盒子高度);行内元素垂直居中 text-align: center;line-height: ( 30 / vw ); 2、盒子居中 情景1&#…...

docker安装-centos

Docker CE 支持 64 位版本 CentOS 7&#xff0c;并且要求内核版本不低于 3.10 卸载旧版本Docker sudo yum remove docker \ docker-common \ docker-selinux \ docker-engine使用yum安装 yum 更新到最新版本: sudo yum update执行以下命令安装依赖包&#xff1a; sudo yum…...

Redis入门指南

文章目录 Redis概述Redis基本数据类型Redis与MySQL的区别以及使用场景如何保持双写一致性&#xff08;缓存一致性&#xff09;1. 延迟双删2. 分布式锁&#xff08;强一致性时使用&#xff09;3. 中间件 Redis持久化机制RDB&#xff08;redis database&#xff09;AOF&#xff0…...

K8s之configMap

1. 概述​ 1.1 什么是configMap​ 1.1 什么是configMap configMap是Kubernetes中的一种资源对象&#xff0c;用于存储配置数据。它可以包含键值对&#xff0c;也可以包含来自文件的配置数据。configMap的作用是将配置数据与应用程序的容器分离&#xff0c;使得配置可以在不重…...

提高 NFS Azure 文件共享性能

本文内容 适用于增加预读大小以提高读取吞吐量Nconnect另请参阅 本文介绍如何提高 NFS Azure 文件共享的性能。 适用于 展开表 文件共享类型SMBNFS标准文件共享 (GPv2)、LRS/ZRS 标准文件共享 (GPv2)、GRS/GZRS 高级文件共享 (FileStorage)、LRS/ZRS 增加预读大…...

【Django-ninja】使用schema

在Django Ninja中&#xff0c;"schema"主要是指帮助描述和规范你的API的工具&#xff0c;以便系统能够自动生成文档并提供验证。通俗地说&#xff0c;它有两个主要作用&#xff1a; API文档生成器&#xff1a; Schema 让 Django Ninja 能够自动生成互动式的API文档。…...

【TCP/IP】用户访问一个购物网站时TCP/IP五层参考模型中每一层的功能

当用户访问一个购物网站时&#xff0c;网络上的每一层都会涉及不同的协议&#xff0c;具体网络模型如下图所示。 以下是每个网络层及其相关的协议示例&#xff1a; 物理层&#xff1a;负责将比特流传输到物理媒介上&#xff0c;例如电缆或无线信号。所以在物理层&#xff0c;可…...

Unity 开发注意事项

1. 空Unity消息 Unity消息被运行时事件调用&#xff0c;即使消息体为空也会被调用。因此&#xff0c;删除空消息避免不必要的处理。 例如&#xff1a; using UnityEngine;class Camera : MonoBehaviour {private void FixedUpdate(){}private void Foo(){} } 应该删除未使用…...

[Unity Sentis] Unity Sentis 详细步骤工作流程

文章目录 1. 导入模型文件支持的模型创建运行时模型导入错误 2. 为模型创建输入将数组转换为张量创建多个输入进行操作 3. 创建一个引擎来运行模型创建一个Worker后端类型 4. 运行模型5. 获取模型的输出获取张量输出多个输出打印输出 1. 导入模型文件 要导入 ONNX 模型文件&am…...

力扣144 二叉树的前序遍历 Java版本

文章目录 题目描述递归方法代码 非递归方法代码 题目描述 给你二叉树的根节点 root &#xff0c;返回它节点值的 前序 遍历。 示例 1&#xff1a; 输入&#xff1a;root [1,null,2,3] 输出&#xff1a;[1,2,3] 示例 2&#xff1a; 输入&#xff1a;root [] 输出&#xf…...

《Vue3 基础知识》 使用 GoGoCod 升级到Vue3+ElementPlus 适配处理

此篇为 《Vue2ElementUI 自动转 Vue3ElementPlus&#xff08;GoGoCode&#xff09;》 的扩展&#xff01; Vue3 适配 Vue3 不兼容适配 Vue 3 迁移指南 在此&#xff0c;本章只讲述项目或组件库中遇到的问题&#xff1b; Vue3 移除 o n &#xff0c; on&#xff0c; on&#…...

c#string方法对比

字符串的截取匹配操作在开发中非常常见&#xff0c;比如下面这个示例&#xff1a;我要匹配查找出来字符串数组中以“abc”开头的字符串并打印&#xff0c;我下面分别用了两种方式实现&#xff0c;代码如下&#xff1a; using System; namespace ConsoleApp23{ class Progra…...

Electron实战(一):环境搭建/Hello World/打包exe

文章目录 Electron安装Node.jsNodeJs推荐配置开始Electron项目创建index.js文件创建src目录运行打包生成exe生成安装包踩坑 下一篇Electron实战(二)&#xff1a;将Node.js和UI能力&#xff08;app/BrowserWindow/dialog)等注入html Electron Electron是一个使用JavaScript, HT…...

【C++】运算符重载详解

&#x1f497;个人主页&#x1f497; ⭐个人专栏——C学习⭐ &#x1f4ab;点击关注&#x1f929;一起学习C语言&#x1f4af;&#x1f4ab; 目录 导读 1. 为什么需要运算符重载 2. 运算符重载概念 3. 运算符重载示例 3.1 运算符重载 3.2 >或<运算符 4. 运算符重…...

评论区功能的简单实现思路

评论区功能是社交类项目中的核心组成部分&#xff0c;它涉及到前端的交云和后端的数据处理。基于你的技术栈&#xff08;前端 Vue3&#xff0c;后端 Java&#xff09;&#xff0c;下面是一个具体的实现思路和数据库设计建议&#xff0c;并探索一下知乎的评论系统。 数据库设计…...

Java自救手册

目录 访问地址 访问地址&#xff0c;发现不通&#xff0c;无法访问&#xff1a; 网络不通一般有两种情况&#xff1a; Maven 拿Maven 拿到Maven以后 Maven单独的报红 Git git注意&#xff1a; 目录 访问地址 访问地址&#xff0c;发现不通&#xff0c;无法访问&…...

ASM-HEMT参数提取和模型验证测试

参数提取程序 直流I-V参数提取 DC模型参数提取流程对于ASM-GaN-HEMT模型可以总结在下图中。 以下步骤描述了该流程&#xff1a; 在模型中设置物理参数&#xff0c;如L&#xff08;沟道长度&#xff09;、W&#xff08;沟道宽度&#xff09;、NF&#xff08;栅指数&#xf…...

浅压缩、深压缩、双引擎、计算机屏幕编码……何去何从?

专业视听领域尤其显示控制和坐席控制领域&#xff0c;最近几年最激动人心的技术&#xff0c;莫过于分布式了。 分布式从推出之日就备受关注&#xff1a;担心稳定性的&#xff0c;质疑同步性能的&#xff0c;怀疑画面质量的…… 诚然&#xff0c;我们在此前见多了带着马赛克的…...

2020年通信工程师初级专业实务真题

文章目录 一、第1章 现代通信网概述&#xff1a;信令网、同步网、管理网。第10章 通信业务&#xff1a;通信产业链&#xff0c;通信终端的分类&#xff0c;通信业务的定义及分类二、第3章 接入网&#xff1a;无线接入网的优点&#xff0c;接入网的接口&#xff08;UNI&#xff…...

用风水学重构代码:周易卦象优化系统架构

在软件工程领域&#xff0c;系统架构的优化与性能调优一直是技术专家们不懈探索的核心课题。传统的优化思路往往聚焦于算法效率、资源调度、中间件配置等技术维度。然而&#xff0c;一个更具颠覆性的视角正在悄然兴起&#xff1a;将源自古老东方智慧的《周易》卦象体系&#xf…...

VR环保学习机|开启沉浸式环保教育新时代

在环保教育不断升级的今天&#xff0c;如何让公众尤其是青少年真正理解环保的重要性、养成绿色习惯&#xff0c;已经成为教育领域的重要课题。传统的图文讲解虽然内容充足&#xff0c;却往往缺少互动性和代入感&#xff0c;难以让学习者产生深刻的记忆。VR环保学习机正是在这样…...

大模型“幻觉“频现?RAG技术如何根治三大痛点,实现精准问答?

文章深入解析了RAG&#xff08;检索增强生成&#xff09;技术的核心原理与实现流程&#xff0c;指出大模型普遍存在的三大缺陷&#xff1a;幻觉现象、知识更新缓慢以及领域知识理解有限。RAG通过结合向量数据库、嵌入模型和大语言模型&#xff0c;实现从外部私有知识库检索信息…...

高效文档下载解决方案:让知识获取不再受阻

高效文档下载解决方案&#xff1a;让知识获取不再受阻 【免费下载链接】kill-doc 看到经常有小伙伴们需要下载一些免费文档&#xff0c;但是相关网站浏览体验不好各种广告&#xff0c;各种登录验证&#xff0c;需要很多步骤才能下载文档&#xff0c;该脚本就是为了解决您的烦恼…...

用Python手把手实现投影梯度下降(PGD):从SVM到LASSO的实战避坑指南

用Python手把手实现投影梯度下降(PGD)&#xff1a;从SVM到LASSO的实战避坑指南 当数据科学家面对带约束的优化问题时&#xff0c;传统梯度下降往往束手无策。投影梯度下降&#xff08;Projected Gradient Descent, PGD&#xff09;就像一位精准的导航员&#xff0c;每次迭代后…...

3大核心能力+2套配置方案:obsidian-i18n终极汉化指南

3大核心能力2套配置方案&#xff1a;obsidian-i18n终极汉化指南 【免费下载链接】obsidian-i18n 项目地址: https://gitcode.com/gh_mirrors/ob/obsidian-i18n 面对全英文的应用界面&#xff0c;你是否曾因语言障碍而错失高效工具&#xff1f;当专业术语晦涩难懂&#…...

JIT加速不生效?你漏掉了这4个强制启用开关,3.14新增--enable-jit-unsafe-mode正在被92%团队忽略

第一章&#xff1a;JIT加速不生效&#xff1f;你漏掉了这4个强制启用开关&#xff0c;3.14新增--enable-jit-unsafe-mode正在被92%团队忽略Go 3.14 引入了激进的 JIT 编译优化路径&#xff0c;但默认关闭全部 JIT 后端。大量团队在升级后观察到 GOMAXPROCS8 下 CPU 利用率未提升…...

d2s-editor:让暗黑破坏神2存档修改变得高效而简单

d2s-editor&#xff1a;让暗黑破坏神2存档修改变得高效而简单 【免费下载链接】d2s-editor 项目地址: https://gitcode.com/gh_mirrors/d2/d2s-editor 在暗黑破坏神2的冒险旅程中&#xff0c;玩家常常面临三大痛点&#xff1a;反复刷装备消耗大量时间、测试不同职业bui…...

ComfyUI-Manager:一站式AI绘画插件智能管理平台

ComfyUI-Manager&#xff1a;一站式AI绘画插件智能管理平台 【免费下载链接】ComfyUI-Manager ComfyUI-Manager is an extension designed to enhance the usability of ComfyUI. It offers management functions to install, remove, disable, and enable various custom node…...

OpenClaw配置备份指南:千问3.5-27B模型迁移与快速恢复

OpenClaw配置备份指南&#xff1a;千问3.5-27B模型迁移与快速恢复 1. 为什么需要备份OpenClaw配置&#xff1f; 上周我的主力开发机突然硬盘故障&#xff0c;导致所有OpenClaw配置丢失。当时正在运行的3个自动化流程全部中断&#xff0c;最棘手的是那个每天凌晨自动整理技术文…...