进程控制(二)进程等待

news2024/11/27 0:44:30

文章目录

  • 进程等待
    • 什么是进程等待???
    • 为什么要进行进程等待???
  • 进程等待的方法
    • wait函数
    • waitpid函数

进程等待

什么是进程等待???

进程等待是通过wait/waitpid的方式,让父进程(一般)对子进程进行资源回收的过程。

为什么要进行进程等待???

1. 为了解决僵尸进程所带来的内存泄漏的问题
在前面的学习中,我们了解到了一种进程状态叫做僵尸状态(子进程终止时,父进程没有对其的资源进行回收)

2. 确认子进程完成工作的进度
父进程创建子进程,让子进程完成某项工作,那么子进程工作完成的进度等信息父进程是一定要知道的,所以需要通过进程等待来获取子进程的退出信息(两个数字)退出信号、退出码。

进程等待的必要性:
如果子进程退出,父进程不管不顾,就可能造成‘僵尸进程’的问题,进而造成内存泄漏。

进程一旦变成僵尸状态,那就刀枪不入,“杀人不眨眼”的kill -9 也无能为力,因为谁也没有办法杀死一个已经死去的进程。

最后,父进程派给子进程的任务完成的如何,我们需要知道。如,子进程运行完成,结果对还是不对,或者是否正常退出。

父进程通过进程等待的方式,回收子进程资源,获取子进程退出信息

进程等待的方法

wait函数

在这里插入图片描述

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


我们通过下面代码来演示wait是如何进行等待的

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

void worker()
{
  int n = 5;
  while(n)
  {
    printf("I am a child process ,pid:%d, ppid:%d, cnt:%d\n", getpid(), getppid(), n);
    n--;
    sleep(1);
  }
}

int main()
{
  pid_t id = fork();
  if(id == 0)
  {
    //child
    worker();                                                                                                                              
    exit(0);
  }
  else
  {
    sleep(10);
    //father
    pid_t rid = wait(NULL);
    if(rid == id)
    {
      printf("wait success \n");
    }
  }
  return 0;
}

在这里插入图片描述
监视一下:while :; do ps axj | head -1 && ps axj | grep test|grep -v grep;echo "---------------"; sleep 1 ;done

在这里插入图片描述

上述代码前五秒钟,父子进程一起运行,其状态为S或R,五秒过后子进程退出但其资源并未回收,所以后面5秒子进程会变成僵尸状态,父进程10s后会打印出上述代码(wait success),打印出来之后也就意味着父进程已经完成了子进程的资源回收,那么子进程的僵尸状态也就消失了。

上述代码证明了:
进程等待能够回收子进程僵尸状态,使其由Z状态变为X状态

wait在等待的时候,父进程又在做什么?

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

void worker()
{
  int n = 5;
  while(n)
  {
    printf("I am a child process ,pid:%d, ppid:%d, cnt:%d\n", getpid(), getppid(), n);
    n--;
    sleep(1);
  }
}

int main()
{
  pid_t id = fork();
  if(id == 0)
  {
    //child
    worker();                                                                                                                              
    exit(0);
  }
  else
  {
    //father
    printf("wait before\n");
    pid_t rid = wait(NULL);
    printf("wait after\n");
    
    if(rid == id)
    {
      printf("wait success \n");
    }
    sleep(10);
  }
  return 0;
}

在这里插入图片描述
在这里插入图片描述

一开始父子进程同时运行,当子进程在运行这5秒期间,父进程是通过调用了wait方法进程阻塞等待的,五秒期间过后才打印出来wait after,说明这段时间父进程是在等子进程变成僵尸状态,wait自动回收

上述代码证明了:
如果子进程没有退出,父进程必须在wait上进行阻塞等待。直到子进程变为僵尸进程,wait自动回收,返回!
一般来说,父子进程谁先运行是不确定的,但一定是父进程最后退出的。

waitpid函数

在这里插入图片描述

返回值:等待成功返回进程pid;等待失败返回 -1;
pid:要等待进程的pid;pid=-1, 等待任一个子进程。与 wait 等效。
status:WIFEXITED(status): 若为正常终止子进程返回的状态,则为真。(查看进程是否是正常退出)
WEXITSTATUS(status): 若 WIFEXITED 非零,提取子进程退出码。(查看进程的退出码)
options:WNOHANG: 若pid指定的子进程没有结束,则waitpid()函数返回0,不予以等待。若正常结束,则返回该子进程的ID。

其中第三个参数设置为0表示的是阻塞等待。

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

void worker()
{
  int n = 5;
  while(n)
  {
    printf("I am a child process ,pid:%d, ppid:%d, cnt:%d\n", getpid(), getppid(), n);
    n--;
    sleep(1);
  }
}
int main()
{
	pid_t id = fork();
	if(id == 0)
	{
		worker();
		exit(10);	
	}
	else
	{
		printf("wait before\n");
		int status = 0;
		pid_t rid = waitpid(id, &status, 0);
		printf("wait after\n");
		if(rid == id)
		{
			printf("wait success\n");
		} 
		
	}
	return 0;
}

waitpid的第二个参数
上面说过进程等待其实是为了解决子进程僵尸状态所造成的内存泄漏问题,但是进程等待还有第二个目的,就是检查子进程的运行结果,即获取子进程的退出码和终止信号。而这个工作就是有waitpid的第二个参数来做的。
在这里插入图片描述
在这里插入图片描述
这个status虽然是一个32位的整型,但我们只使用它的低16位,其中,
低八位存储的是终止信号
次低八位存储的是退出状态

在这里插入图片描述
上面设置的退出状态是exit(10),也就是10,而10用二进制来表示是1010,放在高8位上则是
0000 1010 0000 0000,而该二进程算出来就是我们上面打印出来的status的值:2560.
在这里插入图片描述

退出码:
(status >> 8)&0xFF
退出信号:
status&0x7F

在这里插入图片描述
在这里插入图片描述
exit sig: 0 代表的信号是0,而信号中并没有0信号,说明运行是成功的!
但是exit code:1,代表结果不正确。
运行正常和结果正确必须exit sig和exit code都为0.
但是当我们的exit sig!=0时,就代表代码运行异常,此时exit code则无意义。
在这里插入图片描述


除0错误:
在这里插入图片描述

在这里插入图片描述
exit sig = 8,对应的就是除零错误。


Q:当一个进程异常了,exit code退出码还有意义吗?
A:无意义(你考试已经作弊了,100分和0分有区别么)


如果我们直接在命令行中kill杀死子进程,信号会是什么呢?
在这里插入图片描述
我们发现,使用kill掉子进程后,进程的退出信号就是9,然而当进程由于信号异常终止时,此时进程退出码是无意义的!


waitpid的第三个参数option
option默认为0代表父进程阻塞等待子进程死亡,阻塞等待
就是父进程什么都不干,就在waitpid函数处停下等待子进程变成僵尸进程,然后回收资源。
那么在等待子进程变成僵尸之前,父进程怎样才可以做其他工作,而不是硬等?
非阻塞等待
WNOHANG:等待的时候,以非阻塞的方式等待。

阻塞等待:子进程不退出,wait不返回。
非阻塞等待:往往需要重复调用,轮询+非阻塞,优点:在我们等待子进程变为僵尸的时候,我们也可以做自己的事情,节省时间。

进程的非阻塞式等待:

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

int main()
{
	pid_t pid;
	pid = fork();
	if(pid < 0)
	{
		printf("%s fork error\n",__FUNCTION__);
		return 1;
	}
	else if( pid == 0 )
	{ 	//child
		printf("child is run, pid is : %d\n",getpid());
		sleep(5);
		exit(1);
	} 
	else
	{
		int status = 0;
		pid_t ret = 0;
		do
		{
			ret = waitpid(-1, &status, WNOHANG);//非阻塞式等待
			if( ret == 0 )
			{
				printf("child is running\n");
			}
			sleep(1);
		}while(ret == 0);
		if( WIFEXITED(status) && ret == pid )
		{
		printf("wait child 5s success, child return code is :%d.\n",WEXITSTATUS(status));
		}
		else
		{
			printf("wait child failed, return.\n");
			return 1;
		}
	}
	return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

异或运算实现加密解密

异或运算符^&#xff0c;相同为0&#xff0c;不同为1&#xff08;同0非1&#xff09; 由异或运算法则可知&#xff1a;a ^ a 0&#xff0c;a ^ 0 a 如果c a ^ b&#xff0c;那么a b ^ c&#xff0c;即a ^ b ^ b a&#xff0c;^ 的逆运算仍然是 ^ 利用异或运算的性质&am…

python封装的.exe文件是如何在cmd中获取.xml路径的?

这段日子搞项目算法封装&#xff0c;愁死我。来回改了三遍&#xff0c;总算把相对路径、绝对路径&#xff0c;还有cmd给.exe传参的方式搞懂了。 主要是这个语句 workspace sys.argv[1] sys.argv[]的作用就是,在运行python文件的时候从外部输入参数往文件里面传递参数。 外部就…

09. 配置Eth-Trunk

文章目录 一. 初识Eth-Trunk1.1. Eth-Trunk的概述1.2. Eth-Trunk的优势1.3. Eth-Trunk的模式的优势 二. 实验专题2.1. 实验1&#xff1a;手工模式2.1.1. 实验拓扑图2.1.2. 实验步骤&#xff08;1&#xff09;配置PC机的IP地址&#xff08;2&#xff09;在交换机接口划入VLAN&am…

oracle 21C RAC+linux 8安装配置手册

本文详细介绍利用虚拟机安装配置oracle 21c rac OS版本&#xff1a;oracle linux 8.4 oracle版本&#xff1a;21.3 博主文章目前只发布在两个平台CSDN和墨天伦 ID&#xff1a;潇湘秦&#xff0c;转载请注明出处 安装oracle linux 8.4 选择本地可用的ISO 文件&#xff08;虚拟…

AI Prompt工程师 学习整理

前言 如果说Al大语言模型(LLM,Large Language Model)是宝藏我,那么Prompt提示词就是打开宝藏的钥匙。 最新一代的Al大语言模型具备出色的创作能力,能够生成富有人类感情、严谨逻辑、多场景应用的内容,而如何获得高质量的回答,正确学习使用Prompt提示词是关键。 &#x1f4a5…

vue-cli项目运行流程介绍

一、前言 ​ 本文介绍 vue-cli搭建的项目运行流程&#xff0c;基于已经搭建好的基础项目。关于 vue-cli 构建项目的详细流程&#xff0c;可参考博文&#xff1a;使用vue脚手架构建项目 二、main.js 项目运行 会加载入口文件 main.js /* html文件中&#xff0c;通过script …

MYSQL基础(一) --- 学习笔记

一.基础操作 启动与关闭&#xff1a;net start mysql80&#xff0c;net stop mysql80 客户端连接&#xff1a;mysql -u root -p 关系型数据库&#xff1a;建立在关系模型基础上&#xff0c;由多张相互连接的二维表&#xff08;由行和列组成的表&#xff09;组成的数据库 二…

《Pandas 简易速速上手小册》第6章:Pandas 时间序列分析(2024 最新版)

文章目录 6.1 时间序列数据基础6.1.1 基础知识6.1.2 重点案例&#xff1a;股票市场分析6.1.3 拓展案例一&#xff1a;温度变化分析6.1.4 拓展案例二&#xff1a;电商平台日销售额分析 6.2 日期与时间功能6.2.1 基础知识6.2.2 重点案例&#xff1a;活动日志分析6.2.3 拓展案例一…

最全整理!37 个 Python Web 开发框架总结!

大家好&#xff0c;用了 2 周的时间整理了 Python 中所有的网站开发库&#xff08;下文简称&#xff1a;Web 框架&#xff09;&#xff0c;供大家学习参考。 Q&#xff1a;Web 框架到底是什么&#xff1f; A&#xff1a;Web 框架主要用于网站开发。开发者在基于 Web 框架实现…

【数据结构(C语言)】树、二叉树详解

目录 文章目录 前言 一、树的概念及结构 1.1 树的概念 1.2 树的相关概念 1.3 树的表示 1.4 树在实际中的运用 二、二叉树的概念及结构 2.1 二叉树的概念 2.2 二叉树的基本形态 ​编辑2.3 特殊的二叉树 2.4 二叉树的性质 2.5 二叉树的存储结构 三、二叉树的顺序结…

Java编程练习之类的封装2

1.封装一个股票&#xff08;Stock&#xff09;类&#xff0c;大盘名称为上证A股&#xff0c;前一日的收盘点是2844.70点&#xff0c;设置新的当前值如2910.02点&#xff0c;控制台既要显示以上信息&#xff0c;又要显示涨跌幅度以及点数变化的百分比。运行效果如下&#xff1a;…

SketchBook 2022下载安装教程,保姆级教程,操作简单,小白也能轻松搞定,附安装包和工具

前言 Autodesk SketchBook是一款新一代的自然画图软件&#xff0c;软件界面新颖动人&#xff0c;功能强大&#xff0c;仿手绘效果逼真&#xff0c;笔刷工具分为铅笔&#xff0c;毛笔&#xff0c;马克笔&#xff0c;制图笔&#xff0c;水彩笔&#xff0c;油画笔&#xff0c;喷枪…

Offer必备算法_前缀和_牛客+力扣OJ题详解(由易到难)

目录 前缀和算法介绍 一维前缀和 二维前缀和 ①牛客DP34 【模板】前缀和 解析代码 ②牛客DP35 【模板】二维前缀和 解析代码 ③力扣724. 寻找数组的中心下标 解析代码 ④力扣238. 除自身以外数组的乘积 解析代码 ⑤力扣560. 和为 K 的子数组 解析代码 ⑥力扣974. …

2024美国大学生数学建模竞赛A-F题完整思路+配套代码数据+后续高质量参考论文更新

The Mathematical Contest in Modeling (MCM) The Interdisciplinary Contest in Modeling (ICM) 24美赛【完整每问手把手详细思路可修改50页多种思路版本word版保奖论文】配套升级求解代码可视化图表 美赛A-F题完整版获取见文末 下文包含&#xff1a;2024美国大学生数学建模…

代码随想录 Leetcode110.平衡二叉树

题目&#xff1a; 代码(首刷看解析 2024年1月30日&#xff09;&#xff1a; class Solution { public:int depth(TreeNode* root) {if (root nullptr) return 0;int leftHeight depth(root->left);if (leftHeight -1) return -1;int rightHeight depth(root->right)…

Java List的合并与切分

在Java开发中经常遇到list结构数据的处理&#xff0c;如List的合并或拆分&#xff0c;记录下来&#xff0c;方便备查。 一、List 合并 两个list数据的合并处理&#xff0c;可使用Java8 新特性的stream流&#xff0c;根据实际需要遍历取值。 1、定义 UserInfo 对象 订单的相…

LeetCode: 189.轮转数组

本篇目标了解&#xff0c;翻转数组的经典解法&#xff0c; 189. 轮转数组 - 力扣&#xff08;LeetCode&#xff09; 目录 基本方法概述&#xff1a; 1&#xff0c;翻转做法&#xff0c;推荐时O&#xff08;n&#xff09;&#xff0c;空&#xff08;1&#xff09; 2&#x…

Windows IIS服务如何配置并制作web站点结合内网穿透实现公网访问

文章目录 1. 安装IIS必要WebDav组件2. 客户端测试3. cpolar内网穿透3.1 打开Web-UI管理界面3.2 创建隧道3.3 查看在线隧道列表3.4 浏览器访问测试 4. 安装Raidrive客户端4.1 连接WebDav服务器4.2 连接成功4.2 连接成功总结&#xff1a; 自己用Windows Server搭建了家用NAS主机&…

【Linux C | I/O模型】IO复用 | select、pselect函数详解(看完就会用了)

&#x1f601;博客主页&#x1f601;&#xff1a;&#x1f680;https://blog.csdn.net/wkd_007&#x1f680; &#x1f911;博客内容&#x1f911;&#xff1a;&#x1f36d;嵌入式开发、Linux、C语言、C、数据结构、音视频&#x1f36d; &#x1f923;本文内容&#x1f923;&a…

Unity 模板方法模式(实例详解)

文章目录 简介示例1&#xff1a;游戏关卡流程示例2&#xff1a;测试试卷类示例3&#xff1a;游戏场景构建流程示例4&#xff1a;游戏动画序列示例5&#xff1a;游戏对象初始化过程 简介 Unity中的模板方法模式是一种行为设计模式&#xff0c;它在父类中定义了一个算法的框架&a…