当前位置: 首页 > 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;
}

运行结果如图:

http://www.xdnf.cn/news/987553.html

相关文章:

  • 阿里云unbantu、Flask部署模型的一个错误
  • 安卓+苹果端签名教程
  • SiteAzure:文章删除后,前台还能搜索到
  • HarmonyOS - UIObserver(无感监听)
  • TF-IDF算法的代码实践应用——关键词提取、文本分类、信息检索
  • 帆软 BI 从入门到实战全攻略(一):安装激活与添加数据
  • 大量RPM仓库管理指南:更新与批量获取实战手册
  • VS2017----打开ui文件几秒后闪退
  • 汇编(函数调用)
  • 刷新网站 favicon 的几种方法
  • 医院重症监护系统 ICU重症病房管理系统 重症监护软件
  • QT第一课 —— 设置CMake路径
  • Rust:在Windows上安装
  • BEV和OCC学习-7:mmdet3d 3D检测demo测试
  • 剑指offer21——反转链表
  • 使用html写一个倒计时页面
  • 将模型保存到kaggle中的model中
  • 解码 K-Means 聚类:开启数据星河的炫酷聚类新纪元
  • 前端项目主题切换
  • 解锁Wi-SUN潜能!移远通信发布KCM0A5S模组,点亮智慧城市新图景
  • 关于有害的过度使用 std::move
  • Delphi 获取 XP系统 mac地址
  • Selenium工作原理
  • 【leetcode】36. 有效的数独
  • 利用递归来遍历树
  • Android学习之Window窗口
  • 一个数组样式上要分成两个
  • Unity UGUI GraphicRaycaster.Raycast详解
  • 免费开源的微信开发框架
  • LangSmith 实战指南:大模型链路调试与监控的深度解析