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

《Linux C编程实战》笔记番外:如何避免子进程成为僵尸进程

什么时候会产生僵尸进程?

僵尸进程(Zombie Process) 产生的条件是:

子进程已经结束(退出),但其父进程尚未调用 wait()waitpid() 来回收它的退出信息(即子进程的退出状态)。

  • 在这种情况下,子进程的进程表项仍保留,以便父进程将来可以获取其退出状态。

  • 这个进程已经“死了”,但还在操作系统中“留个壳子”,这就是“僵尸”。

🔍 具体过程:

  1. 子进程调用 exit() 或正常返回结束;

  2. 内核将其状态设为 ZOMBIE(僵尸);

  3. 等待父进程来调用 wait()waitpid()

  4. 父进程调用 wait() 后,内核清理子进程的 PCB(进程控制块),从而完全释放其资源。

父进程结束会不会结束子进程?

不会。父进程退出时,子进程不会自动结束,而是会被init 进程(PID 1)接管(在 Linux 中)。

这类子进程称为孤儿进程,不会变成僵尸,因为 init 进程会自动为其调用 wait(),避免产生僵尸。

避免僵尸进程的方法

子进程终止时将产生SIGCHLD信号,知道这一点就很容易完成消灭僵尸进程。

  1. 在父进程中使用 signal(SIGCHLD, SIG_IGN);

    • 表示父进程忽略子进程退出信号,内核将自动回收子进程资源,无需显式调用 wait()

  2. 使用 fork() 后立即 wait() 或异步 waitpid() 轮询

 本文章将采用方法1的方式,用信号处理子进程退出信号,并且采用自己设定的信号处理函数而不是SIG_IGN来展示

 基础知识在以下文章,请自行学习:

《Linux C编程实战》笔记:进程操作之创建进程_linux用c代码创建实时进程-CSDN博客

《Linux C编程实战》笔记:进程操作之退出,执行,等待_linux c exec程序-CSDN博客 

《Linux C编程实战》笔记:信号的捕捉和处理_c语言信号捕捉与利用-CSDN博客 

 测试代码如下:

#include <cstdio>
#include <iostream>
#include<sys/types.h>
#include<sys/wait.h>
#include<unistd.h>
#include<signal.h>
void read_childproc(int sig) {//信号处理函数int status;pid_t id = waitpid(-1, &status, WNOHANG);//非阻塞等待子进程结束;-1表示等待任意子进程//判断是否是正常结束的,正常结束WIFEXITED返回非0,进入if语句if (WIFEXITED(status)) {printf("Removed proc id:%d\n", id);printf("Child send:%d\n", WEXITSTATUS(status));//该宏返回子进程的退出状态}
}
int main(int argc,char *argv[])
{pid_t pid;//使用sigaction方法注册信号处理struct sigaction act;act.sa_handler = read_childproc;//信号处理函数sigemptyset(&act.sa_mask);//无屏蔽码act.sa_flags = 0;sigaction(SIGCHLD, &act, 0);pid = fork();//创建子进程if (pid == -1) {//创建失败perror("Process creation failed\n");exit(EXIT_FAILURE);}else if (pid == 0) {//子进程std::cout << "Hi! I'm child process" << std::endl;sleep(10);return 12;//子进程返回值}else {//父进程std::cout << "Child proc id:" << pid << std::endl;pid = fork();//再次创建一个子进程if (pid == -1) {//创建失败perror("Process creation failed\n");exit(EXIT_FAILURE);}else if (pid == 0) {//子进程std::cout << "Hi! I'm child process" << std::endl;sleep(10);exit(24);//子进程返回值}else {//父进程std::cout << "Child proc id:" << pid << std::endl;for (int i = 0; i < 5; i++) {//睡五次五秒。因为发生信号时,父进程会被唤醒,所以暂停时间不到25秒std::cout << "wait..." << std::endl;sleep(5);}}}return 0;
}

运行结果如图:

相关文章:

  • 每日Prompt:人像写真
  • uni-app bitmap.load() 返回 code=-100
  • xilinx的gt的ALIGN_COMMA_WORD设置的作用
  • 鸿蒙新闻应用全链路优化实践:从内核重构到体验革新
  • 实时获取印度国家股票数据 API 实操
  • Explore Image Deblurring via Encoded Blur Kernel Space论文阅读
  • AIStor 的模型上下文协议 (MCP) 服务器: 工作原理
  • [Git] 配置 Git
  • 3D IC(立体集成电路)的生态机会
  • 多项目资源如何高效配置与再分配?
  • 【JavaAPI搜索引擎】项目测试报告
  • JSON 是什么?核心概念与语法解析
  • 神经网络:深度剖析过拟合、欠拟合及其泛化能力提升策略
  • 计算机视觉与深度学习 | 基于Matlab的低照度图像增强算法:全面总结与实现
  • K8S多维度问题排查
  • 安卓中草药宝典 V2.6上千种中草药知识学习
  • K8S项目需求分析
  • Android11 Launcher3实现去掉抽屉改为单层
  • GO 入门小项目-博客-结合Gin Gorm
  • AWS之混合云
  • 如何做网站seo韩小培/网站优化 推广
  • wordpress 如何从新安装/云优客seo排名公司
  • 网站推广实践内容/百度网页游戏
  • 丽水网站建设/国内最新十大新闻
  • 粉丝网站制作/百度搜索引擎怎么做
  • 云主机如何建网站/东莞网站建设最牛