【Linux C | 进程】进程终止、等待 | exit、_exit、wait、waitpid

😁博客主页😁:🚀https://blog.csdn.net/wkd_007🚀
🤑博客内容🤑:🍭嵌入式开发、Linux、C语言、C++、数据结构、音视频🍭
🤣本文内容🤣:🍭介绍 🍭
😎金句分享😎:🍭你不能选择最好的,但最好的会来选择你——泰戈尔🍭

本文未经允许,不得转发!!!

目录

  • 🎄一、进程终止
    • ✨1.1 正常终止
    • ✨1.2 异常终止
  • 🎄二、孤儿进程、僵死进程
    • ✨2.1 孤儿进程
    • ✨2.2 僵死进程
  • 🎄三、等待子进程结束 | wait、waitpid
    • ✨3.1 wait 函数
    • ✨3.2 waitpid 函数
  • 🎄四、总结


在这里插入图片描述

🎄一、进程终止

✨1.1 正常终止

进程有下面5种正常终止方式:

  • 1、在main函数内执行return语句。这等效于调用exit。
    在子函数中调用return不会导致进程退出,而调用exit会导致整个进程退出。

    #include <stdio.h>
    #include <stdlib.h>
    int child_fun()
    {//return 0;	// 不导致进程退出exit(0);	// 会导致进程退出
    }
    int main()
    {child_fun();while(1)sleep(1);exit(0);	// 等效于 return 0;
    }
    
  • 2、调用exit函数。此函数由ISO C定义,其操作包括调用各终止处理程序(终止处理程序在调用atexit函数时登记),然后关闭所有标准IO流等。因为ISO C并不处理文件描述符、多进程(父、子进程)以及作业控制,所以这定义对UNIX系统而言是不完整的。

    #include <stdio.h>
    #include <stdlib.h>
    #include <unistd.h>
    void end_deal()
    {printf("atexit deal ...\n");
    }
    int main()
    {printf("程序开始...\n");printf("调用 atexit 注册\n");atexit(end_deal);printf("程序结束\n");exit(0);
    }
    

    在这里插入图片描述

  • 3、调用_exit_Exit函数。ISO C定义_Exit,其目的是为进程提供一种无需运行终止处理程序或信号处理程序而终止的方法。对标准IO流是否进行冲洗,这取决于实现。在UNIX系统中,_Exit_exit是同义的,并不清洗标准I/O流。_exit函数由exit调用,它处理UNIX特定的细节。_exit是由POSIX.1说明的。

    #include <stdio.h>
    #include <stdlib.h>
    #include <unistd.h>
    void end_deal()
    {printf("atexit deal ...\n");
    }
    int main()
    {printf("程序开始...\n");printf("调用 atexit 注册\n");atexit(end_deal);printf("程序结束\n");_exit(0);
    }
    

    _exit退出进程时,不会调用atexit注册的函数。
    在这里插入图片描述

  • 4、进程的最后一个线程在其启动例程中执行返回语句。但是,该线程的返回值不会用作进程的返回值。当最后一个线程从其启动例程返回时,该进程以终止状态0返回。

  • 5、进程的最后一个线程调用pthread_exit函数。如同前面一样,在这种情况中,进程终止状态总是0,这与传送给pthread_exit的参数无关。


✨1.2 异常终止

三种异常终止方式如下:

  • 1、调用abort。它产生SIGABRT信号,这是下一种异常终止的一种特例
  • 2、当进程接收到某些信号时。信号可由进程自身(例如调用abort函数)、其他进程或内核产生。例如,若进程越出其地址空间访问存储单元或者除以0,内核就会为该进程产生相应的信号。
  • 3、最后一个线程对“取消”(cancellation)请求做出响应。按系统默认,“取消”以延迟方式发生:一个线程要求取消另一个线程,一段时间之后,目标线程终止。

不管进程如何终止,最后都会执行内核中的同一段代码。这段代码为相应进程关闭所有打开描述符,释放它所使用的存储器等。

终止进程时,我们希望能够通知其父进程它时如何终止的。对于三个终止函数((exit、_exit和_Exit),实现这一点的方法是,将其退出状态(exit status)作为参数传送给函数。在异常终止情况下,内核(不是进程本身)产生一个指示其异常终止原因的终止状态(termination status)。在任意一种情况下,该终止进程的父进程都能用waitwaitpid函数取得其终止状态。


在这里插入图片描述

🎄二、孤儿进程、僵死进程

✨2.1 孤儿进程

父进程先于子进程结束,子进程变成孤儿进程,该孤儿进程会被1号进程(init进程)领养

其操作过程大致如下:在一个进程终止时,内核逐个检查所有活动进程,以判断它是否是正要终止进程的子进程,如果是,则将该进程的父进程ID更改为1 (init进程的ID)。这种处理方法保证了每个进程都有一个父进程。

#include <stdio.h>
#include <unistd.h>int main()
{printf("父进程%d开始运行,并创建子进程\n", getpid());pid_t pid = fork();if(pid == 0){printf("子进程%d被创建并开始运行,父进程是%d\n",getpid(),getppid());printf("子进程进入睡眠状态");sleep(3);printf("子进程睡醒后,发现父进程变为%d\n",getppid());return 0;}printf("父进程继续运行....\n");sleep(1);printf("父进程执行结束\n");return 0;
}

运行结果如下:
在这里插入图片描述


✨2.2 僵死进程

在UNIX术语中,一个已经终止、但是其父进程尚未对其进行善后处理(获取终止子进程的有关信息,释放它仍占用的资源)的进程被称为僵死进程(zombie)。ps(1)命令将僵死进程的状态打印为Z。如果编写一个长期运行的程序,它调用fork产生了很多子进程,那么除非父进程等待取得子进程的终止状态,否则这些子进程终止后就会变成僵死进程。

如果子进程先结束,会给父进程发信号,如果父进程没有收到该信号或没有及时处理,子进程将变成僵死进程,直到父进程处理了该信号或者父进程退出。

内核为每个终止子进程保存了一定量的信息,所以当终止进程的父进程调用waitwaitpid时,可以得到这些信息。这些信息至少包括进程ID、该进程的终止状态、以及该进程使用的CPU时间总量。

#include <stdio.h>
#include <unistd.h>int main()
{printf("%d 进程:我要调用fork()了...\n", getpid());pid_t pid = fork();if(pid == 0){printf("%d进程:我是%d的子进程,即将成为僵尸...\n",getpid(),getppid());return 0;//子进程结束,自动发信号}sleep(1);printf("%d进程: 我是%d的父进程\n", getpid(),pid);getchar();return 0;
}

运行结果打印:
在这里插入图片描述
打开另一个命令行窗口,查看子进程状态为僵死进程:
在这里插入图片描述


在这里插入图片描述

🎄三、等待子进程结束 | wait、waitpid

本节介绍两个函数(wait、waitpid),用于等待调用进程的子进程中的状态更改,并获取子进程状态更改的信息。
状态变化被认为是:①子进程被终止;②子进程被一个信号拦住了;③子进程被一个信号恢复了。
在终止子进程的情况下,执行wait函数允许系统释放与该子进程相关联的资源;如果不执行等待,则终止的子进程将保持“僵尸”状态。

下表是判断退出状态的宏

解释
WIFEXITED(status)若为正常终止子进程返回的状态,则为真。对于这种情况可执行WEXITSTATUS( status),取子进程传送给exit、_exit或_Exit参数的低8位
WEXITSTATUS(status)可以获取退出码
WIFSIGNALED(status )若为异常终止子进程返回的状态,则为真(接到一个不捕捉的信号)。对于这种情况,可执行WTERMSIG(status),取使子进程终止的信号编号。另外,有些实现(非Single UNIX Specification)定义宏WCOREDUMP status),若已产生终止进程的core文件,则它返回真
WIFSTOPPED(siatus)若为当前暂停子进程的返回的状态,则为真。对于这种情况,可执行wSTOPSIG(status),取使子进程暂停的信号编号
WIFCONTINUED(status)若在作业控制暂停后已经继续的子进程返回了状态,则为真。(POSIX.1的XSI扩展,仅用于waitpid。)

✨3.1 wait 函数

函数原型:

#include <sys/wait.h>
pid_t wait(int *status);
  • 功能:wait函数会使调用进程一直阻塞,直到它的一个子进程终止为止,它等效于:waitpid(-1, &status, 0);。如果调用进程没有子进程则立即返回。
  • 参数:status,是传出参数,用来带出结束子进程的退出码和退出状态;
  • 返回值:成功返回终止的子进程的进程ID,失败返回 -1

看例子:

#include <stdio.h>
#include <stdlib.h>
#include <sys/wait.h>
int main()
{pid_t pid = fork();if(pid == 0){printf("子进程开始执行,即将睡眠...\n");sleep(3);printf("子进程运行结束\n");exit(132);//该值不应该超过255}//父进程printf("父进程开始执行,子进程的PID=%d\n",pid);printf("父进程等待子进程结束....\n");int status = 0;/*如果子进程不退出,该函数一直阻塞*/pid_t res = wait(&status);printf("发现%d子进程结束", res);printf("status = %d\n", status);/*如果子进程正常结束,该宏的操作结果为1,反之为0*/if(WIFEXITED(status)){printf("子进程正常结束\n");printf("返回值为:%d\n", WEXITSTATUS(status));}return 0;
}

运行结果:
在这里插入图片描述


✨3.2 waitpid 函数

函数原型:

#include <sys/wait.h>
pid_t waitpid(pid_t pid, int *status, int options);
  • 功能:waitpid函数会使调用进程一直阻塞,直到指定的子进程终止为止。

  • 参数:

    • pid:
      <-1:等待进程组ID等于pid绝对值的任何子进程;
      -1:等待任意子进程,与wait等效;
      0:等待进程组ID等于调用进程的进程组ID的任何子进程;
      >0:等待进程ID等于pid值的子进程。

    • status:是传出参数,用来带出结束子进程的退出码和退出状态;

    • options:
      0:调用进程阻塞等待;
      WNOHANG:如果没有子进程终止,则立即返回。

  • 返回值:成功时,返回状态已更改的子进程的进程ID;如果指定了WNOHANG,并且pid指定的一个或多个子级存在,但尚未更改状态,则返回0。出现错误时,返回-1。

看例子:

#include <stdio.h>
#include <stdlib.h>
#include <sys/wait.h>
int main()
{int pid1 = 0;int pid2 = 0;pid1 = fork();if(pid1>0){pid2 = fork();}if(pid1 == 0)//子进程一{printf("子进程一开始执行 PID=%d,睡3秒\n",getpid());sleep(3);printf("子进程一结束\n");exit(55);}if(pid2 == 0)//子进程二{printf("子进程二开始执行 PID=%d,睡1秒\n",getpid());sleep(1);printf("子进程二结束\n");exit(200);}//父进程printf("父进程等待子进程一结束\n");int status = 0;/*阻塞等待子进程一结束*/waitpid(pid1, &status, 0);if(WIFEXITED(status)){printf("子进程一正常结束,返回值:%d\n",WEXITSTATUS(status));}printf("父进程结束\n");
}

运行结果:
在这里插入图片描述


在这里插入图片描述

🎄四、总结

本文先介绍了进程终止的8个方式,然后介绍孤儿进程、僵死进程,最后介绍了父进程等待子进程的两个函数wait、waitpid。

在这里插入图片描述
如果文章有帮助的话,点赞👍、收藏⭐,支持一波,谢谢 😁😁😁

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

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

相关文章

【Kafka】开发实战和Springboot集成kafka

目录 消息的发送与接收生产者消费者 SpringBoot 集成kafka服务端参数配置 消息的发送与接收 生产者 生产者主要的对象有&#xff1a; KafkaProducer &#xff0c; ProducerRecord 。 其中 KafkaProducer 是用于发送消息的类&#xff0c; ProducerRecord 类用于封装Kafka的消息…

仅使用K-M法+Cox比例风险模型就能发二区文章 | SEER公共数据库周报(1.17)

欢迎各位参加本周中山大学著名卫生统计学家方积乾教授公益直播讲座&#xff01; 就在本周三晚&#xff0c;主题为“真实世界研究与RCT研究”&#xff0c;欢迎各位预约参加&#xff01; SEER&#xff08;The Surveillance, Epidemiology, and End Results&#xff09;数据库是由…

回溯算法篇-01:全排列

力扣46&#xff1a;全排列 题目分析 这道题属于上一篇——“回溯算法解题框架与思路”中的 “元素不重复不可复用” 那一类中的 排列类问题。 我们来回顾一下当时是怎么说的&#xff1a; 排列和组合的区别在于&#xff0c;排列对“顺序”有要求。比如 [1,2] 和 [2,1] 是两个不…

柔性数组和C语言内存划分

柔性数组和C语言内存划分 1. 柔性数组1.1 柔性数组的特点&#xff1a;1.2 柔性数组的使用1.3 柔性数组的优势 2. 总结C/C中程序内存区域划分 1. 柔性数组 也许你从来没有听说过柔性数组&#xff08;flexible array)这个概念&#xff0c;但是它确实是存在的。 C99 中&#xff…

力扣740. 删除并获得点数

动态规划 思路&#xff1a; 选择元素 x&#xff0c;获得其点数&#xff0c;删除 x 1 和 x - 1&#xff0c;则其他的 x 的点数也会被获得&#xff1b;可以将数组转换成一个有序 map&#xff0c;key 为 x&#xff0c; value 为对应所有 x 的和&#xff1b;则问题转换成了不能同…

Postman基本使用、测试环境(Environment)配置

文章目录 准备测试项目DemoController测试代码Interceptor模拟拦截配置 Postman模块简单介绍Postman通用环境配置新建环境(Environment)配置环境(Environment)设置域名变量引用域名变量查看请求结果打印 Postman脚本设置变量登录成功后设置全局Auth-Token脚本编写脚本查看conso…

即插即用篇 | UniRepLKNet:用于音频、视频、点云、时间序列和图像识别的通用感知大卷积神经网络 | DRepConv

大卷积神经网络(ConvNets)近来受到了广泛研究关注,但存在两个未解决且需要进一步研究的关键问题。1)现有大卷积神经网络的架构主要遵循传统ConvNets或变压器的设计原则,而针对大卷积神经网络的架构设计仍未得到解决。2)随着变压器在多个领域的主导地位,有待研究ConvNets…

软件设计师——项目管理(一)

&#x1f4d1;前言 本文主要是【项目管理】——软件设计师——项目管理的文章&#xff0c;如果有什么需要改进的地方还请大佬指出⛺️ &#x1f3ac;作者简介&#xff1a;大家好&#xff0c;我是听风与他&#x1f947; ☁️博客首页&#xff1a;CSDN主页听风与他 &#x1f304…

Databend 开源周报第 129 期

Databend 是一款现代云数仓。专为弹性和高效设计&#xff0c;为您的大规模分析需求保驾护航。自由且开源。即刻体验云服务&#xff1a;https://app.databend.cn 。 Whats On In Databend 探索 Databend 本周新进展&#xff0c;遇到更贴近你心意的 Databend 。 支持标准流 标…

如何在 Ubuntu 22.04 上安装 Apache Web 服务器

前些天发现了一个人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;最重要的屌图甚多&#xff0c;忍不住分享一下给大家。点击跳转到网站。 如何在 Ubuntu 22.04 上安装 Apache Web 服务器 介绍 Apache HTTP 服务器是世界上使用最广泛的 Web 服务器。它…

模拟器单窗口ip有问题?试试关闭IPV6来解决

目前应该不止雷电9有这个问题了&#xff0c;最早是看到无忧群里在说有这个问题&#xff0c;后面发现很多其他的ip软件也有同样的问题&#xff0c;很多人都遇到&#xff0c;所以做个图文教程在这里&#xff0c;没出问题的也可以设置一下&#xff0c;目前ipv6也还没普及&#xff…

x-cmd pkg | hurl - HTTP 请求处理工具

目录 简介首次用户功能特点竞品和相关作品进一步探索 简介 Hurl 是 HTTP 请求处理工具&#xff0c;支持使用简单的纯文本格式定义的 HTTP 请求。它的用途非常广泛&#xff0c;既可以用于获取数据&#xff0c;也可以用于测试HTTP会话。 它可以链式处理请求&#xff0c;捕获数值…

ORA-01033: ORACLE initialization or shutdown in progress---惜分飞

客户反馈数据库使用plsql dev登录报ORA-01033: ORACLE initialization or shutdown in progress的错误 出现该错误一般是由于数据库没有正常open成功,查看oracle 告警日志发现 Mon Jan 22 16:55:50 2024 Database mounted in Exclusive Mode Lost write protection disabled …

Unity SRP 管线【第五讲:URP烘培光照】

本节&#xff0c;我们将跟随数据流向讲解UEP管线中的烘培光照。 文章目录 一、URP烘培光照1. 搭建场景2. 烘培光照参数设置MixedLight光照设置&#xff1a;直观感受 Lightmapping Settings参数设置&#xff1a; 3. 我们如何记录次表面光源颜色首先我们提取出相关URP代码&#…

企业数字档案馆的构成要素

企业数字档案馆的构成要素包括以下几个方面&#xff1a; 1. 系统平台&#xff1a;企业数字档案馆需要有一个稳定的系统平台&#xff0c;用于存储、管理和检索档案信息。这个平台可以是基于云计算、数据库或其他技术的&#xff0c;能够支持大容量的数据存储和快速的检索功能。 2…

设计模式二(工厂模式)

本质&#xff1a;实例化对象不用new&#xff0c;用工厂代替&#xff0c;实现了创建者和调用者分离 满足&#xff1a; 开闭原则&#xff1a;对拓展开放&#xff0c;对修改关闭 依赖倒置原则&#xff1a;要针对接口编程 迪米特原则&#xff1a;最少了解原则&#xff0c;只与自己直…

Unity—配置lua环境变量+VSCode 搭建 Lua 开发环境

每日一句&#xff1a;保持须臾的浪漫&#xff0c;理想的喧嚣&#xff0c;平等的热情 Windows 11下配置lua环境变量 一、lua-5.4.4版本安装到本地电脑 链接&#xff1a;https://pan.baidu.com/s/14pAlOjhzz2_jmvpRZf9u6Q?pwdhd4s 提取码&#xff1a;hd4s 二、高级系统设置 此电…

P9232 [蓝桥杯 2023 省 A] 更小的数

[蓝桥杯 2023 省 A] 更小的数 终于本弱一次通关了一道研究生组别的题了[普及/提高−] 一道较为简单的双指针题,但一定有更好的解法. 题目描述 小蓝有一个长度均为 n n n 且仅由数字字符 0 ∼ 9 0 \sim 9 0∼9 组成的字符串&#xff0c;下标从 0 0 0 到 n − 1 n-1 n−1&a…

C++ //练习 2.35 判断下列定义推断出的类型是什么,然后编写程序进行验证。

C Primer&#xff08;第5版&#xff09; 练习 2.35 练习 2.35 判断下列定义推断出的类型是什么&#xff0c;然后编写程序进行验证。 const int i 42; auto j i; const auto &k i; auto *p &i; const auto j2 i, &k2 i;环境&#xff1a;Linux Ubuntu&#x…

HarmonyOS鸿蒙学习基础篇 - 运行第一个程序 Hello World

下载与安装DevEco Studio 古话说得好&#xff0c;“磨刀不误砍柴工”&#xff0c;对于HarmonyOS应用开发&#xff0c;我们首先得确保工具齐全。这就好比要进行HarmonyOS应用开发&#xff0c;我们需要确保已经安装了DevEco Studio&#xff0c;这是HarmonyOS的一站式集成开发环境…