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

Linux笔记---进程:进程等待

1. 进程等待的概念

进程等待是指父进程通过系统调用waitwaitpid来对子进程进行状态检测与回收的功能。

当子进程退出时,如果父进程不读取子进程的退出状态,子进程就会成为僵尸进程,造成内存泄漏的问题。因此,父进程需要调用wait或者waitpid确认子进程的退出信息以回收僵尸进程的资源以防止内存泄漏。

进程等待的必要性

  1. 回收僵尸进程:子进程退出后,如果父进程不进行处理,子进程会变成僵尸进程,占用系统资源,可能导致内存泄漏。
  2. 获取子进程的退出情况:父进程可以通过进程等待获取子进程的退出码和退出信号,从而了解子进程的执行结果

 2. 进程等待的方法

进程等待通过包含在头文件<sys/types.h>和<sys/wait.h>中的两个函数实现:wait()waitpid()

wait和waitpid函数用于等待子进程,等待意味着等待为子进程收尸,父进程可能会在wait或waitpid处阻塞,等待子进程退出。

2.1 wait函数

pid_t wait(int* status);
  • 函数原型pid_t wait(int *status);
  • 功能:使调用的进程(通常是父进程)暂停执行,直到一个子进程终止或发生一个信号。
  • 参数status是一个输出型参数,用于存放子进程的终止状态,如果不关心子进程的退出状态,可以设置为NULL。
  • 返回值:如果有子进程退出,wait()返回子进程的PID,并可通过status指针获取子进程的退出状态;如果等待失败,则返回-1。

 2.2 waitpid函数

pid_ t waitpid(pid_t pid, int *status, int options);
  • 函数原型pid_t waitpid(pid_t pid, int *status, int options);
  • 功能:提供更多的控制,允许父进程等待特定的子进程,或者是与父进程有特定关系的任何子进程。
  • 参数
    • pid:指定要等待的子进程的PID;若为-1,则等待任何子进程,与wait等效。
    • status:和wait一样,用于存放子进程的终止状态。
    • options:可以控制waitpid的行为,该参数选择性传入,不传时行为与wait相同。
      • WNOHANG:如果指定的子进程没有结束,waitpid函数不会阻塞(进程不会暂停等待子进程结束),而是立即返回0。
      • WUNTRACED:返回终止子进程信息和因信号停止的子进程信息。
      • WCONTINUED:返回收到SIGCONT信号而恢复执行的已停止子进程状态信息
  • 返回值
    • 当正常返回的时候,waitpid返回收集到的子进程的进程ID。
    • 如果设置了选项WNOHANG,而调用中waitpid发现没有已退出的子进程可收集,则返回0。
    • 如果调用中出错,则返回-1,这时errno会被设置成相应的值以指示错误所在。

 2.3 status的用法

*status作为一个整型值,其四个字节的各个字段分别用于存储各类信息,我们可以将其看作位图。

我们并不需要记住哪些字段用于存储什么信息,如何解析,因为操作系统为我们定义了一系列的宏来对*status进行解析:

  1. WIFEXITED(status)

    • 功能:检查子进程是否正常退出。
    • 返回值:如果子进程通过调用exit_exit正常退出,则返回一个非零值。
    • 使用示例:
      if (WIFEXITED(status)) {printf("子进程正常退出\n");
      }
  2. WEXITSTATUS(status)

    • 功能:获取子进程的退出状态码。
    • 返回值:返回子进程通过exit_exit系统调用设置的退出状态码。
    • 使用示例:
      if (WIFEXITED(status)) {int exit_status = WEXITSTATUS(status);printf("子进程的退出状态码是:%d\n", exit_status);
      }
  3. WIFSIGNALED(status)

    • 功能:检查子进程是否因为接收到信号而异常终止。
    • 返回值:如果子进程因为接收到信号而终止,则返回一个非零值。
    • 使用示例:
      if (WIFSIGNALED(status)) {printf("子进程因为信号而异常终止\n");
      }
  4. WTERMSIG(status)

    • 功能:获取导致子进程终止的信号编号。
    • 返回值:返回导致子进程终止的信号编号。
    • 使用示例:
      if (WIFSIGNALED(status)) {int signal_number = WTERMSIG(status);printf("导致子进程终止的信号编号是:%d\n", signal_number);
      }
  5. WIFSTOPPED(status)

    • 功能:检查子进程是否因为接收到信号而暂停。
    • 返回值:如果子进程因为接收到信号而暂停,则返回一个非零值。
    • 使用示例:
      if (WIFSTOPPED(status)) {printf("子进程因为信号而暂停\n");
      }
  6. WSTOPSIG(status)

    • 功能:获取导致子进程暂停的信号编号。
    • 返回值:返回导致子进程暂停的信号编号。
    • 使用示例:
      if (WIFSTOPPED(status)) {int stop_signal = WSTOPSIG(status);printf("导致子进程暂停的信号编号是:%d\n", stop_signal);
      }

http://www.lryc.cn/news/495275.html

相关文章:

  • 【Linux】匿名管道通信场景——进程池
  • 算法妙妙屋-------1.递归的深邃回响:全排列的奇妙组合
  • 【maven-6】Maven 生命周期相关命令演示
  • 黑马程序员Java笔记整理(day06)
  • LeetCode【代码随想录】刷题(动态规划篇)
  • 【看海的算法日记✨优选篇✨】第三回:二分之妙,寻径中道
  • 基于yolov8、yolov5的铝材缺陷检测识别系统(含UI界面、训练好的模型、Python代码、数据集)
  • 计算机光电成像理论基础
  • 【QNX+Android虚拟化方案】125 - 如何创建android-spare镜像
  • 深度学习基础小结_项目实战:手机价格预测
  • EMall实践DDD模拟电商系统总结
  • 【随笔】AI技术在电商中的应用
  • 序列式容器详细攻略(vector、list)C++
  • 快速启动项目
  • springboot347基于web的铁路订票管理系统(论文+源码)_kaic
  • 使用API管理Dynadot域名,在账户中添加域名服务器(Name Server)
  • 【Linux | 计网】TCP协议深度解析:从连接管理到流量控制与滑动窗口
  • go语言的成神之路-筑基篇-对文件的操作
  • 两道数据结构编程题
  • 【Qt】QDateTimeEdit控件实现清空(不保留默认时间/最小时间)
  • 12、字符串
  • DPDK用户态协议栈-Tcp Posix API 1
  • 【人工智能-科普】图神经网络(GNN):与传统神经网络的区别与优势
  • LabVIEW实现UDP通信
  • [pdf,epub]228页《分析模式》漫谈合集01-45提供下载
  • Kafka的消费消息是如何传递的?
  • 二分查找(Java实现)(1)
  • 力扣103.二叉树的锯齿形层序遍历
  • Search with Orama
  • 一万台服务器用saltstack还是ansible?