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

Linux系统中处理子进程的终止问题

1. 理解子进程终止的机制

在Unix/Linux系统中,当子进程终止时,会向父进程发送一个SIGCHLD信号。父进程需要捕捉这个信号,并通过调用wait()waitpid()等函数来回收子进程的资源。这一过程被称为“回收僵尸进程”。

如果父进程没有及时调用wait()或相关函数,子进程将会成为僵尸进程,占用系统资源,直到父进程终止或调用相应的等待函数。

2. 使用wait()waitpid()函数

  • wait():使父进程阻塞,直到任一子进程终止。它会返回终止子进程的PID,并存储子进程的退出状态。
  • waitpid():提供更精细的控制,可以等待特定的子进程或采用非阻塞方式。
例子:阻塞等待子进程终止
#include <iostream>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>int main() {pid_t pid = fork();  // 创建子进程if (pid < 0) {std::cerr << "Fork failed!" << std::endl;return 1;}else if (pid == 0) {// 子进程执行的代码std::cout << "子进程PID:" << getpid() << " 启动。" << std::endl;sleep(2);  // 模拟子进程工作std::cout << "子进程PID:" << getpid() << " 结束。" << std::endl;return 42;  // 子进程以状态42退出}else {// 父进程执行的代码std::cout << "父进程PID:" << getpid() << " 等待子进程结束。" << std::endl;int status;pid_t terminated_pid = wait(&status);  // 阻塞等待任一子进程结束if (terminated_pid > 0) {if (WIFEXITED(status)) {std::cout << "子进程PID:" << terminated_pid << " 以状态 " << WEXITSTATUS(status) << " 退出。" << std::endl;}else if (WIFSIGNALED(status)) {std::cout << "子进程PID:" << terminated_pid << " 被信号 " << WTERMSIG(status) << " 终止。" << std::endl;}}else {std::cerr << "等待子进程失败。" << std::endl;}}return 0;
}

输出示例:

父进程PID:12345 等待子进程结束。
子进程PID:12346 启动。
子进程PID:12346 结束。
子进程PID:12346 以状态 42 退出。

代码解释:

  1. 父进程调用fork()创建子进程。
  2. 子进程执行自己的任务后,以状态42退出。
  3. 父进程调用wait()阻塞等待子进程结束,并获取子进程的退出状态。
  4. 父进程通过宏WIFEXITEDWEXITSTATUS判断子进程是否正常退出及其退出状态。

3. 使用SIGCHLD信号处理异步回收

为了避免父进程被阻塞,可以通过信号处理函数异步处理子进程的终止。这在需要父进程继续执行其他任务时非常有用。

例子:使用SIGCHLD处理子进程终止
#include <iostream>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <signal.h>
#include <cstring>// 信号处理函数
void sigchld_handler(int signum) {// 循环回收所有已终止的子进程while (true) {int status;pid_t pid = waitpid(-1, &status, WNOHANG);if (pid <= 0) {break;}if (WIFEXITED(status)) {std::cout << "[Signal Handler] 子进程PID:" << pid << " 以状态 " << WEXITSTATUS(status) << " 退出。" << std::endl;}else if (WIFSIGNALED(status)) {std::cout << "[Signal Handler] 子进程PID:" << pid << " 被信号 " << WTERMSIG(status) << " 终止。" << std::endl;}}
}int main() {// 注册SIGCHLD信号处理函数struct sigaction sa;sa.sa_handler = sigchld_handler;sigemptyset(&sa.sa_mask);sa.sa_flags = SA_RESTART | SA_NOCLDSTOP;if (sigaction(SIGCHLD, &sa, NULL) == -1) {std::cerr << "无法注册SIGCHLD处理器:" << strerror(errno) << std::endl;return 1;}pid_t pid = fork();  // 创建子进程if (pid < 0) {std::cerr << "Fork failed!" << std::endl;return 1;}else if (pid == 0) {// 子进程执行的代码std::cout << "子进程PID:" << getpid() << " 启动。" << std::endl;sleep(2);  // 模拟子进程工作std::cout << "子进程PID:" << getpid() << " 结束。" << std::endl;return 24;  // 子进程以状态24退出}else {// 父进程执行的其他任务std::cout << "父进程PID:" << getpid() << " 正在执行其他任务。" << std::endl;// 模拟父进程执行其他任务for (int i = 0; i < 5; ++i) {std::cout << "父进程执行中:" << i + 1 << std::endl;sleep(1);}// 父进程结束前确保所有子进程已被回收// 可以调用wait(NULL)或者让信号处理器完成回收}return 0;
}

输出示例:

父进程PID:12345 正在执行其他任务。
子进程PID:12346 启动。
父进程执行中:1
父进程执行中:2
子进程PID:12346 结束。
[Signal Handler] 子进程PID:12346 以状态 24 退出。
父进程执行中:3
父进程执行中:4
父进程执行中:5

代码解释:

  1. 注册SIGCHLD信号处理器
    • 使用sigaction结构体注册sigchld_handler函数作为SIGCHLD信号的处理器。
    • SA_RESTART标志用于在信号处理后自动重启被中断的系统调用。
    • SA_NOCLDSTOP标志表示当子进程停止或继续时,父进程不接收SIGCHLD信号。
  2. 创建子进程
    • 子进程执行自己的任务并以状态24退出。
  3. 父进程执行其他任务
    • 父进程在等待子进程结束的同时,继续执行其他任务,不会被阻塞。
  4. 信号处理函数sigchld_handler
    • 当子进程终止时,SIGCHLD信号会被触发,sigchld_handler函数会被调用。
    • 在函数内部,使用waitpidWNOHANG选项非阻塞地回收所有已终止的子进程,防止僵尸进程的产生。

4. 避免僵尸进程的策略

  • 及时调用wait()waitpid():确保父进程在子进程终止后,立即回收其资源。
  • 使用信号处理器:如上文所示,通过注册SIGCHLD信号处理器,可以在子进程终止时自动回收资源,而不需要父进程主动等待。
  • 设置SIGCHLDSIG_IGN:在一些系统上,可以通过将SIGCHLD信号的处理方式设置为忽略,从而自动回收子进程资源。这种方法不适用于所有情况,需谨慎使用。
例子:设置SIGCHLD为忽略
#include <iostream>
#include <unistd.h>
#include <signal.h>
#include <cstring>int main() {// 设置SIGCHLD为忽略signal(SIGCHLD, SIG_IGN);pid_t pid = fork();  // 创建子进程if (pid < 0) {std::cerr << "Fork failed!" << std::endl;return 1;}else if (pid == 0) {// 子进程执行的代码std::cout << "子进程PID:" << getpid() << " 启动。" << std::endl;sleep(2);  // 模拟子进程工作std::cout << "子进程PID:" << getpid() << " 结束。" << std::endl;return 0;}else {// 父进程执行其他任务std::cout << "父进程PID:" << getpid() << " 正在执行其他任务。" << std::endl;sleep(5);  // 父进程等待子进程结束}return 0;
}

注意事项:

  • 这种方法依赖于系统对SIGCHLD的具体实现,不保证在所有Unix/Linux系统中都有效。
  • 尽管简便,但可能无法获取子进程的退出状态,限制了错误处理和日志记录的能力。

5. 处理多个子进程的终止

当父进程创建多个子进程时,需要确保所有子进程的终止都被正确处理,以避免僵尸进程。可以在SIGCHLD处理函数中使用循环调用waitpid,直到所有终止的子进程都被回收。

例子:处理多个子进程
#include <iostream>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <signal.h>
#include <vector>
#include <cstring>// 信号处理函数
void sigchld_handler(int signum) {// 循环回收所有已终止的子进程while (true) {int status;pid_t pid = waitpid(-1, &status, WNOHANG);if (pid <= 0) {break;}if (WIFEXITED(status)) {std::cout << "[Signal Handler] 子进程PID:" << pid << " 以状态 " << WEXITSTATUS(status) << " 退出。" << std::endl;}else if (WIFSIGNALED(status)) {std::cout << "[Signal Handler] 子进程PID:" << pid << " 被信号 " << WTERMSIG(status) << " 终止。" << std::endl;}}
}int main() {// 注册SIGCHLD信号处理函数struct sigaction sa;sa.sa_handler = sigchld_handler;sigemptyset(&sa.sa_mask);sa.sa_flags = SA_RESTART | SA_NOCLDSTOP;if (sigaction(SIGCHLD, &sa, NULL) == -1) {std::cerr << "无法注册SIGCHLD处理器:" << strerror(errno) << std::endl;return 1;}std::vector<pid_t> child_pids;// 创建多个子进程for (int i = 0; i < 3; ++i) {pid_t pid = fork();if (pid < 0) {std::cerr << "Fork failed!" << std::endl;return 1;}else if (pid == 0) {// 子进程执行的代码std::cout << "子进程PID:" << getpid() << " 启动。" << std::endl;sleep(2 + i);  // 模拟不同的工作时间std::cout << "子进程PID:" << getpid() << " 结束。" << std::endl;return i;}else {// 父进程记录子进程PIDchild_pids.push_back(pid);}}// 父进程执行其他任务std::cout << "父进程PID:" << getpid() << " 正在执行其他任务。" << std::endl;sleep(6);  // 等待所有子进程结束// 由于SIGCHLD处理器已经回收了子进程,父进程无需再次调用waitreturn 0;
}

输出:

父进程PID:12345 正在执行其他任务。
子进程PID:12346 启动。
子进程PID:12347 启动。
子进程PID:12348 启动。
子进程PID:12346 结束。
[Signal Handler] 子进程PID:12346 以状态 0 退出。
子进程PID:12347 结束。
[Signal Handler] 子进程PID:12347 以状态 1 退出。
子进程PID:12348 结束。
[Signal Handler] 子进程PID:12348 以状态 2 退出。

代码解释:

  1. 创建多个子进程:循环调用fork()创建三个子进程,每个子进程有不同的工作时间。
  2. 信号处理函数sigchld_handler会被多次调用,以回收每个子进程的资源。
  3. 父进程执行其他任务:父进程在子进程运行期间继续执行其他任务,不会被阻塞。
  4. 无需显式等待:由于信号处理器已经负责回收子进程,父进程无需再次调用wait()waitpid()

6. 使用prctl设置子进程终止时的行为

在某些情况下,可以通过prctl系统调用设置子进程终止时的行为。例如,可以设置PR_SET_CHILD_SUBREAPER,使得特定的进程成为“子进程的收割者”,用于复杂的进程管理。

例子:设置PR_SET_CHILD_SUBREAPER
#include <iostream>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <signal.h>
#include <cstring>
#include <sys/prctl.h>int main() {// 将当前进程设置为子进程的收割者if (prctl(PR_SET_CHILD_SUBREAPER, 1) == -1) {std::cerr << "prctl failed: " << strerror(errno) << std::endl;return 1;}pid_t pid = fork();if (pid < 0) {std::cerr << "Fork failed!" << std::endl;return 1;}else if (pid == 0) {// 子进程执行的代码std::cout << "子进程PID:" << getpid() << " 启动。" << std::endl;sleep(2);std::cout << "子进程PID:" << getpid() << " 结束。" << std::endl;return 0;}else {// 父进程作为子进程的收割者,等待子进程结束std::cout << "父进程设置为子进程的收割者,等待子进程结束。" << std::endl;int status;pid_t terminated_pid = wait(&status);if (terminated_pid > 0) {if (WIFEXITED(status)) {std::cout << "子进程PID:" << terminated_pid << " 以状态 " << WEXITSTATUS(status) << " 退出。" << std::endl;}}}return 0;
}

代码解释:

  • prctl(PR_SET_CHILD_SUBREAPER, 1):将当前进程设置为子进程的收割者,使其能够回收其子孙进程的资源。
  • 这种设置在需要复杂的子进程管理,或在容器化环境中非常有用。

相关文章:

Linux系统中处理子进程的终止问题

1. 理解子进程终止的机制 在Unix/Linux系统中&#xff0c;当子进程终止时&#xff0c;会向父进程发送一个SIGCHLD信号。父进程需要捕捉这个信号&#xff0c;并通过调用wait()或waitpid()等函数来回收子进程的资源。这一过程被称为“回收僵尸进程”。 如果父进程没有及时调用w…...

Docker 不再难懂:快速掌握容器命令与架构原理

1. Docker 是容器技术的一种 容器&#xff08;Container&#xff09;概述 容器&#xff08;Container&#xff09;是一种轻量级的虚拟化技术&#xff0c;它将应用程序及其所有依赖环境打包在一个独立的、可移植的运行时环境中。容器通过操作系统级的虚拟化提供隔离&#xff0…...

取消票证会把指定的票证从数据库中删除,同时也会把票证和航班 等相关表中的关联关系一起删除。但在删除之前,它会先检查当前用户是否拥有这张票

在做航班智能客服问答系统时会遇到取消票证的场景&#xff0c;这里涉及数据库的操作时会把指定的票证从数据库中删除&#xff0c;同时也会把票证和航班等相关表中的关联关系一起删除。但在删除之前&#xff0c;需要先检查当前用户是否拥有这张票&#xff0c;只有票主才有权限取…...

力扣-贪心-763 划分字母区间

思路 先统计字符串中每一个字母出现的最后下标&#xff0c;然后从end初始化为第一个字母出现的最后下标&#xff0c;在i<end时&#xff0c;不断更新end&#xff0c;因为一旦囊括新的字母就最起码要遍历到新字母出现的最后下标&#xff0c;在i>end时&#xff0c;说明遍历…...

【Redis 原理】网络模型

文章目录 用户空间 && 内核空间阻塞IO非阻塞IO信号驱动IO异步IOIO多路复用selectpollepoll Web服务流程Redis 网络模型Redis单线程网络模型的整个流程Redis多线程网络模型的整个流程 用户空间 && 内核空间 为了避免用户应用导致冲突甚至内核崩溃&#xff0c;用…...

cpp中的继承

一、继承概念 在cpp中&#xff0c;封装、继承、多态是面向对象的三大特性。这里的继承就是允许已经存在的类&#xff08;也就是基类&#xff09;的基础上创建新类&#xff08;派生类或者子类&#xff09;&#xff0c;从而实现代码的复用。 如上图所示&#xff0c;Person是基类&…...

DeepSeek全栈接入指南:从零到生产环境的深度实践

第一章:DeepSeek技术体系全景解析 1.1 认知DeepSeek技术生态 DeepSeek作为新一代人工智能技术平台,构建了覆盖算法开发、模型训练、服务部署的全链路技术栈。其核心能力体现在: 1.1.1 多模态智能引擎 自然语言处理:支持文本生成(NLG)、语义理解(NLU)、情感分析等计算…...

CSS 真的会阻塞文档解析吗?

在网页开发领域&#xff0c;一个常见的疑问是 CSS 是否会阻塞文档解析。理解这一问题对于优化网页性能、提升用户体验至关重要。要深入解答这个问题&#xff0c;需要从浏览器渲染网页的原理说起。 浏览器渲染网页的基本流程 浏览器在接收到 HTML 文档后&#xff0c;会依次进行…...

大模型的UI自动化:Cline 使用Playwright MCP Server完成测试

大模型的UI自动化:Cline 使用Playwright MCP Server完成测试 MCP MCP(Model Context Protocol),是一个开发的协议,标准化了应用程序如何为大模型提供上下文。MCP提供了一个标准的为LLM提供数据、工具的方式,使用MCP会更容易的构建Agent或者是基于LLM的复杂工作流。 最近…...

碰撞检测 | 图解凸多边形分离轴定理(附ROS C++可视化)

目录 0 专栏介绍1 凸多边形碰撞检测2 多边形判凸算法3 分离轴定理(SAT)4 算法仿真与可视化4.1 核心算法4.2 仿真实验 0 专栏介绍 &#x1f525;课设、毕设、创新竞赛必备&#xff01;&#x1f525;本专栏涉及更高阶的运动规划算法轨迹优化实战&#xff0c;包括&#xff1a;曲线…...

Python 基本数据类型

目录 1. 字符串&#xff08;String&#xff09; 2. 列表&#xff08;List&#xff09; 3. 字典&#xff08;Dictionary&#xff09; 4. 集合&#xff08;Set&#xff09; 5. 数字&#xff08;Number&#xff09; 6. 布尔值&#xff08;Boolean&#xff09; 1. 字符串&…...

突破“第一崇拜“:五维心理重构之路

一、视频介绍 在这个崇尚"第一"的时代&#xff0c;我们如何找到自己的独特价值&#xff1f;本视频将带您踏上五维心理重构之旅&#xff0c;从诗意人生的角度探讨如何突破"圣人之下皆蝼蚁"的局限。我们将穿越人生的不同阶段&#xff0c;从青春的意气风发到…...

KubeKey一键安装部署k8s集群和KubeSphere详细教程

目录 一、KubeKey简介 二、k8s集群KubeSphere安装 集群规划 硬件要求 Kubernetes支持版本 操作系统要求 SSH免密登录 配置集群时钟 所有节点安装依赖 安装docker DNS要求 存储要求 下载 KubeKey 验证KubeKey 配置集群文件 安装集群 验证命令 登录页面 一、Ku…...

UE5网络通信架构解析

文章目录 前言一、客户端-服务器架构&#xff08;C/S Model&#xff09;二、对等网络架构&#xff08;P2P&#xff0c;非原生支持&#xff09;三、混合架构&#xff08;自定义扩展&#xff09;四、UE5网络核心机制 前言 UE5的网络通信主要基于客户端-服务器&#xff08;C/S&am…...

实验3 知识表示与推理

实验3 知识表示与推理 一、实验目的 &#xff08;1&#xff09;掌握知识和知识表示的基本概念&#xff0c;理解其在AI中的深刻含义与意义&#xff1b; &#xff08;2&#xff09;熟悉AI中常用的知识表示方法的优缺点及其应用场景&#xff1b; &#xff08;3&#xff09;掌握产…...

基于Springboot银行信用卡额度管理系统【附源码】

基于Springboot银行信用卡额度管理系统 效果如下&#xff1a; 系统登陆页面 用户个人中心页面 新增信用卡申请页面 评估审核页面 管理员主页面 评估审核页面 操作日志管理页面 消费页面 研究背景 随着金融行业的快速发展和信息技术的不断进步&#xff0c;信用卡作为一种便捷…...

达梦数据库学习笔记@1

目录 达梦数据库学习笔记一、表空间管理&#xff08;一&#xff09;默认表空间&#xff08;二&#xff09;相关数据字典&#xff08;三&#xff09;表空间操作&#xff08;四&#xff09;临时表空间管理 二、重做日志管理&#xff08;一&#xff09;系统视图&#xff08;二&…...

图像处理篇---图像处理中常见参数

文章目录 前言一、分贝&#xff08;dB&#xff09;的原理1.公式 二、峰值信噪比&#xff08;PSNR, Peak Signal-to-Noise Ratio&#xff09;1.用途2.公式3.示例 三、信噪比&#xff08;SNR, Signal-to-Noise Ratio&#xff09;1.用途2.公式3.示例 四、动态范围&#xff08;Dyna…...

AI Agent实战:打造京东广告主的超级助手 | 京东零售技术实践

前言 自2022年末ChatGPT的问世&#xff0c;大语言模型&#xff08;LLM&#xff09;技术引发全球关注。在大模型技术落地的最佳实践中&#xff0c;智能体&#xff08;Agent&#xff09;架构显现出巨大潜力&#xff0c;成为业界的普遍共识&#xff0c;各大公司也纷纷启动Agent技…...

50周学习go语言:第1周 环境搭建

以下是为零基础学习者准备的详细第1周教程&#xff0c;包含环境搭建、工具配置和首个Go程序的完整操作指南&#xff1a; 一、Go语言环境安装&#xff08;Windows/macOS/Linux通用&#xff09; 1. 下载安装包 官网地址&#xff1a;https://go.dev/dl//根据系统选择对应版本&am…...

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…...

零门槛NAS搭建:WinNAS如何让普通电脑秒变私有云?

一、核心优势&#xff1a;专为Windows用户设计的极简NAS WinNAS由深圳耘想存储科技开发&#xff0c;是一款收费低廉但功能全面的Windows NAS工具&#xff0c;主打“无学习成本部署” 。与其他NAS软件相比&#xff0c;其优势在于&#xff1a; 无需硬件改造&#xff1a;将任意W…...

(二)TensorRT-LLM | 模型导出(v0.20.0rc3)

0. 概述 上一节 对安装和使用有个基本介绍。根据这个 issue 的描述&#xff0c;后续 TensorRT-LLM 团队可能更专注于更新和维护 pytorch backend。但 tensorrt backend 作为先前一直开发的工作&#xff0c;其中包含了大量可以学习的地方。本文主要看看它导出模型的部分&#x…...

Python实现prophet 理论及参数优化

文章目录 Prophet理论及模型参数介绍Python代码完整实现prophet 添加外部数据进行模型优化 之前初步学习prophet的时候&#xff0c;写过一篇简单实现&#xff0c;后期随着对该模型的深入研究&#xff0c;本次记录涉及到prophet 的公式以及参数调优&#xff0c;从公式可以更直观…...

安宝特案例丨Vuzix AR智能眼镜集成专业软件,助力卢森堡医院药房转型,赢得辉瑞创新奖

在Vuzix M400 AR智能眼镜的助力下&#xff0c;卢森堡罗伯特舒曼医院&#xff08;the Robert Schuman Hospitals, HRS&#xff09;凭借在无菌制剂生产流程中引入增强现实技术&#xff08;AR&#xff09;创新项目&#xff0c;荣获了2024年6月7日由卢森堡医院药剂师协会&#xff0…...

【MATLAB代码】基于最大相关熵准则(MCC)的三维鲁棒卡尔曼滤波算法(MCC-KF),附源代码|订阅专栏后可直接查看

文章所述的代码实现了基于最大相关熵准则(MCC)的三维鲁棒卡尔曼滤波算法(MCC-KF),针对传感器观测数据中存在的脉冲型异常噪声问题,通过非线性加权机制提升滤波器的抗干扰能力。代码通过对比传统KF与MCC-KF在含异常值场景下的表现,验证了后者在状态估计鲁棒性方面的显著优…...

TSN交换机正在重构工业网络,PROFINET和EtherCAT会被取代吗?

在工业自动化持续演进的今天&#xff0c;通信网络的角色正变得愈发关键。 2025年6月6日&#xff0c;为期三天的华南国际工业博览会在深圳国际会展中心&#xff08;宝安&#xff09;圆满落幕。作为国内工业通信领域的技术型企业&#xff0c;光路科技&#xff08;Fiberroad&…...

[论文阅读]TrustRAG: Enhancing Robustness and Trustworthiness in RAG

TrustRAG: Enhancing Robustness and Trustworthiness in RAG [2501.00879] TrustRAG: Enhancing Robustness and Trustworthiness in Retrieval-Augmented Generation 代码&#xff1a;HuichiZhou/TrustRAG: Code for "TrustRAG: Enhancing Robustness and Trustworthin…...

TJCTF 2025

还以为是天津的。这个比较容易&#xff0c;虽然绕了点弯&#xff0c;可还是把CP AK了&#xff0c;不过我会的别人也会&#xff0c;还是没啥名次。记录一下吧。 Crypto bacon-bits with open(flag.txt) as f: flag f.read().strip() with open(text.txt) as t: text t.read…...

相关类相关的可视化图像总结

目录 一、散点图 二、气泡图 三、相关图 四、热力图 五、二维密度图 六、多模态二维密度图 七、雷达图 八、桑基图 九、总结 一、散点图 特点 通过点的位置展示两个连续变量之间的关系&#xff0c;可直观判断线性相关、非线性相关或无相关关系&#xff0c;点的分布密…...