Linux操作系统5- 补充知识(可重入函数,volatile关键字,SIGCHLD信号)
上篇文章:Linux操作系统5-进程信号3(信号的捕捉流程,信号集,sigaction)-CSDN博客
本篇Gitee仓库:myLerningCode/l26 · 橘子真甜/Linux操作系统与网络编程学习 - 码云 - 开源中国 (gitee.com)
目录
一. 可重入函数
二. volatile关键字
2.1 volatile关键字作用
2.2 中断程序下volatile关键字作用
三. SIGCHILD 信号
3.1 使用SIGCHLID处理进程退出
3.2 SIG_IGN清理僵尸进程
一. 可重入函数
我们一般将 main函数执行流和 信号捕捉执行流是两个执行流
有一个函数fun:
如果在main执行流和信号捕捉执行流中,这个函数被重复进入,如果出现了问题 - 则这个函数是不可重入函数。
如果在main执行流和信号捕捉执行流中,这个函数被重复进入,如果没有出现问题 - 则这个函数是可重入函数。
不可重入函数如果在多个执行流中执行的话,可能会导致数据不安全问题
是否可重入是一个中性的形容词。
一般来说:调用了malloc/free/new/delete等的函数是不可重入的 。调用了 I/O操作的函数也是不可重入的。
二. volatile关键字
2.1 volatile关键字作用
vlolatile关键字的作用是:保证某变量的内存可见性。
被vlolatile修饰的变量,系统总是从内存中读取这个数据。此时编译器不会对这个变量过度优化(比如将其写入到某一个寄存器中,从寄存器读取数据)。
2.2 中断程序下volatile关键字作用
下面这段代码可能会出现编译器过度优化而导致的错误。
#include <stdio.h>
#include <unistd.h>
#include <signal.h>int quit = 0;void handler(int signo)
{printf("捕捉到[%d]信号\n", signo);printf("quit: %d", quit);quit = 1;printf("-> %d\n", quit);
}int main()
{signal(2, handler);while (!quit);printf("由于信号捕捉, quit为1 正常退出\n");return 0;
}
该代码收到2号信号就会更改quit,然后进程就能正常结束
不优化情况下编译运行:

可以看到程序没有任何问题。
现在增加编译器的优化, 修改makefile 增加 -O3

可以看到,ctrl c 之后进程收到2号信号将quit改为1,但是不会退出 。
为什么不会退出呢?数据会保存在内存或者寄存器
不优化:执行handler之后,将quit写回内存,main函数退出
优化:quit在main函数执行流没有被修改,编译器之后将其保存在寄存器中,且之后一直从寄存器中读取数据。执行handler之后,将物理内存中的quit改为1,但是寄存器中的数据没有修改,所以进程不会退出。
这样就导致了代码无问题,程序有问题
如果对quit使用volatile关键字修饰,此时会保证其内存可见性(每次读取数据的时候都去内存中读取数据) 。之后就不会出问题了
#include <stdio.h>
#include <unistd.h>
#include <signal.h>volatile int quit = 0;void handler(int signo)
{printf("捕捉到[%d]信号\n", signo);printf("quit: %d", quit);quit = 1;printf("-> %d\n", quit);
}int main()
{signal(2, handler);while (!quit);printf("由于信号捕捉, quit为1 正常退出\n");return 0;
}
运行结果如下:

三. SIGCHILD 信号
3.1 使用SIGCHLID处理进程退出
SIGCHLID(17号)这个信号是子进程退出的时候会向父进程发送这个信号。
我们在进程控制时候提到,可以使用wait或者waitpid来获取子进程退出的信息。处理僵尸进程(子进程退出,父进程太忙无时间处理子进程退出)问题。
我们可以采用阻塞或者非阻塞的方式来等待子进程退出。采用阻塞的话,父进程就不能进行自己的工作。采用非阻塞的方式的话,进行自己工作的时候询问也会降低效率。
进程等待可看这篇文章:Linux操作系统2-进程控制2(进程等待,waitpid系统调用,阻塞与非阻塞等待)_wait系统调用-CSDN博客
我们使用子进程退出,父进程接收17号信号的特点。17号信号的默认行为是忽略,如果我们自定义17号信号的行为,在handler中进行等待子进程退出,这样就能让父进程去执行自己的代码而不用去浪费时间关心子进程退出了。
即子进程退出,向父进程发送SIGCHILD信号,父进程接收信号后调用wait/waitpid清理子进程退出信息。清理僵尸进程。
测试代码如下:

如果有多个子进程同时退出,或者部分子进程退出,需要循环等待它们,并且只要没等待成功就立即结束信号捕捉方法。
代码如下:
#include <iostream>#include <unistd.h>
#include <wait.h>
#include <signal.h>
#include <sys/types.h>void handler(int signo)
{// 处理子进程退出信息std::cout << "我是父进程 pid为:" << getpid() << "收到信号:" << signo << std::endl;while (1){int status = 0;pid_t sid = waitpid(-1, &status, 0);printf("子进程退出码[%d], 子进程退出信号[%d]\n", ((status >> 8) & 0xff), (status & 0x7f));if (sid <= 0)break;}
}int main()
{// 1.自定义SIGCHILD行为signal(SIGCHLD, handler);// 2.创建子进程pid_t id = fork();if (id == 0){// 子进程int cnt = 5;while (cnt--){std::cout << "我是子进程 pid为:" << getpid() << std::endl;sleep(1);}exit(2);}// 父进程int cnt = 0;while (1){std::cout << "我是父进程 pid为:" << getpid() << " 次数为:" << cnt++ << std::endl;sleep(1);}return 0;
}
3.2 SIG_IGN清理僵尸进程
使用SIG_IGN清理僵尸是由于uinx的历史原因。在linux中保留了这种方式。
我们使用signal或者sigaction将SIGCHILD信号的捕捉方法设置为SIG_IGN即可自动帮助我们清理僵尸进程。
测试代码如下:
#include <iostream>#include <unistd.h>
#include <wait.h>
#include <signal.h>
#include <sys/types.h>int main()
{// 1.自定义SIGCHILD行为,并用SIG_IGN清理僵尸进程signal(SIGCHLD, SIG_IGN);// 2.创建子进程pid_t id = fork();if (id == 0){// 子进程int cnt = 5;while (cnt--){std::cout << "我是子进程 pid为:" << getpid() << std::endl;sleep(1);}exit(2);}// 父进程不等待子进程int cnt = 0;while (1){sleep(1);}return 0;
}
测试僵尸进程有没有被处理。
如果我们注释掉 signal(SIGCHLD, SIG_IGN); 这条代码。
运行结果如下:

子进程没有受到处理变为僵尸进程,导致内存泄漏
相关文章:
Linux操作系统5- 补充知识(可重入函数,volatile关键字,SIGCHLD信号)
上篇文章:Linux操作系统5-进程信号3(信号的捕捉流程,信号集,sigaction)-CSDN博客 本篇Gitee仓库:myLerningCode/l26 橘子真甜/Linux操作系统与网络编程学习 - 码云 - 开源中国 (gitee.com) 目录 一. 可重入…...
ctfshow刷题笔记—栈溢出—pwn61~pwn64
目录 前言 一、pwn61(输出了什么?) 二、pwn62(短了一点) 三、pwn63(又短了一点) 四、pwn64(有时候开启某种保护并不代表这条路不通) 五、一些shellcode 前言 这几道都是与shellcode有关的题,实在是…...
java23种设计模式-责任链模式
责任链模式(Chain of Responsibility Pattern)学习笔记 编程相关书籍分享:https://blog.csdn.net/weixin_47763579/article/details/145855793 DeepSeek使用技巧pdf资料分享:https://blog.csdn.net/weixin_47763579/article/details/145884039 🌟 模式定义 责任链模式是…...
新一代跨境电商ERP系统:从订单到发货的全流程自动化管理
随着全球电商市场的持续扩张,跨境电商卖家面临着多平台运营、国际物流、税务合规等复杂挑战。如何高效整合订单、库存、物流和财务数据,实现从客户下单到商品交付的无缝衔接,成为企业降本增效的关键。Zoho Books作为一款专为跨境商家设计的智…...
苹果廉价机型 iPhone 16e 影像系统深度解析
【人像拍摄差异】 尽管iPhone 16e支持后期焦点调整功能,但用户无法像iPhone 16系列那样通过点击屏幕实时切换拍摄主体。前置摄像头同样缺失人像深度控制功能,不过TrueTone原彩闪光灯系统在前后摄均有保留。 很多人都高估了 iPhone 的安全性,查…...
hive 面试题
Hive基础概念 1.1 Hive是什么? 基于Hadoop的数据仓库工具,支持类SQL(HiveQL)查询,底层转换为MapReduce/Tez/Spark任务。 核心功能:数据ETL、查询、分析;定位:OLAP(分析…...
VScode在windows10上使用clang-format
用途:自动调整代码格式,如缩进等。 clang-format官方文档:ClangFormat — Clang 21.0.0git documentation 前提:有一个.clang-format文件 下载LLVM:https://github.com/llvm/llvm-project/releases,将可…...
AWS API Gateway灰度验证实现
在微服务架构中,灰度发布(金丝雀发布)是验证新版本稳定性的核心手段。通过将小部分流量(如 10%)导向新版本服务,可以在不影响整体系统的情况下快速发现问题。AWS API Gateway 原生支持流量按比例分配功能,无需复杂编码即可实现灰度验证。本文将详细解析其实现方法、最佳…...
【每日八股】MySQL篇(三):索引(上)
目录 MySQL 为什么使用 B 树来做索引,它的优势是什么?特性和定义B 树和 B 树的对比拓展:既然 B 树相较于 B 树优势如此之大,为什么 nosql 的 MongoDB 底层仍采用 B 树而不是 B 树? 使用 B 树做索引的优势补充ÿ…...
在Pycharm中将ui文件修改为py文件
在Pycharm中将ui文件修改为py文件 有些时候,我们需要把QTDesigner生成的.ui文件修改为.py文件 在一些教程中,通常使用cmd打开终端修改,或者是有一些人写了一些脚本来修改 这里我们可以使用pycharm来快速的修改 首先,我们在pyc…...
看视频学习方法总结
以下是提高教学视频吸收率的系统性方法,结合认知科学原理和实际学习场景,帮助您最大化学习效果: 一、观看前的黄金准备阶段 60秒快速扫描法 用1分钟快速浏览视频目录、章节标题和简介,建立知识框架。荷兰伊拉斯姆斯大学实验表明&…...
Matlab 大量接单
分享一个matlab接私活、兼职的平台 1、技术方向满足任一即可 2、技术要求 3、最后 技术方向满足即可 MATLAB:熟练掌握MATLAB编程语言,能够使用MATLAB进行数据处理、机器学习和深度学习等相关工作。 机器学习、深度学习、强化学习、仿真、复现、算法、…...
《深度剖析:生成对抗网络中生成器与判别器的高效协作之道》
在人工智能的前沿领域,生成对抗网络(GAN)以其独特的对抗学习机制,为数据生成和处理带来了革命性的变革。生成器与判别器作为GAN的核心组件,它们之间的协作效率直接决定了GAN在图像生成、数据增强、风格迁移等众多应用中…...
Android6到Android15版本新增的功能和api
Android6到Android15版本新增的功能和api 文章目录 Android6到Android15版本新增的功能和api一、前言二、Android6 后的版本迭代1、Android 6.0(Marshmallow,API 级别 23)新增功能重要 API 2、Android 7.0(Nougat,API …...
【现代Web布局与动画技术:卡片组件实战分享】
📱 现代Web布局与动画技术:卡片组件实战分享 🚀 引言 🌟 在过去的开发过程中,我们共同实现了一个功能丰富的卡片组件,它不仅美观,还具有交互性和响应式设计。这篇文章将分享这个组件背后的技术…...
计算机网络之传输层(传输层提供的服务)
一、可靠的数据传输 传输层提供可靠的数据传输服务,确保数据在传输过程中不丢失、不重复、不乱序,并且能够被正确接收。这通常通过面向连接的协议(如TCP)来实现,TCP通过确认、重传、序号等机制来保证数据传输的可靠性…...
FPGA开发,使用Deepseek V3还是R1(1):应用场景
以下都是Deepseek生成的答案 FPGA开发,使用Deepseek V3还是R1(1):应用场景 FPGA开发,使用Deepseek V3还是R1(2):V3和R1的区别 FPGA开发,使用Deepseek V3还是R1&#x…...
哈希表和STL —— unorderde_set/unordered_map【复习笔记】
1. 哈希表的相关概念 1.1 哈希表的定义 哈希表,又称为散列表,是根据关键字直接进行访问的数据结构。 它通过一个哈希函数(Hash Function),建立了一种关键字和存储地址间的直接映射关系,将每个关键字映射…...
计算机毕业设计SpringBoot+Vue.js体育馆使用预约平台(源码+文档+PPT+讲解)
温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 作者简介:Java领…...
42 session反序列化漏洞
参考资料:3. php反序列化从入门到放弃(入门篇) - bmjoker - 博客园 session文件上传漏洞利用原理 当在php.ini中设置session.upload_progress.enabled On的时候,PHP将能够跟踪上传单个文件的上传进度。当上传正在进行时,以及在将与session…...
使用van-uploader 的UI组件,结合vue2如何实现图片上传组件的封装
以下是基于 vant-ui(适配 Vue2 版本 )实现截图中照片上传预览、删除功能,并封装成可复用组件的完整代码,包含样式和逻辑实现,可直接在 Vue2 项目中使用: 1. 封装的图片上传组件 ImageUploader.vue <te…...
Python爬虫(二):爬虫完整流程
爬虫完整流程详解(7大核心步骤实战技巧) 一、爬虫完整工作流程 以下是爬虫开发的完整流程,我将结合具体技术点和实战经验展开说明: 1. 目标分析与前期准备 网站技术分析: 使用浏览器开发者工具(F12&…...
Java多线程实现之Thread类深度解析
Java多线程实现之Thread类深度解析 一、多线程基础概念1.1 什么是线程1.2 多线程的优势1.3 Java多线程模型 二、Thread类的基本结构与构造函数2.1 Thread类的继承关系2.2 构造函数 三、创建和启动线程3.1 继承Thread类创建线程3.2 实现Runnable接口创建线程 四、Thread类的核心…...
AI书签管理工具开发全记录(十九):嵌入资源处理
1.前言 📝 在上一篇文章中,我们完成了书签的导入导出功能。本篇文章我们研究如何处理嵌入资源,方便后续将资源打包到一个可执行文件中。 2.embed介绍 🎯 Go 1.16 引入了革命性的 embed 包,彻底改变了静态资源管理的…...
Unity | AmplifyShaderEditor插件基础(第七集:平面波动shader)
目录 一、👋🏻前言 二、😈sinx波动的基本原理 三、😈波动起来 1.sinx节点介绍 2.vertexPosition 3.集成Vector3 a.节点Append b.连起来 4.波动起来 a.波动的原理 b.时间节点 c.sinx的处理 四、🌊波动优化…...
JAVA后端开发——多租户
数据隔离是多租户系统中的核心概念,确保一个租户(在这个系统中可能是一个公司或一个独立的客户)的数据对其他租户是不可见的。在 RuoYi 框架(您当前项目所使用的基础框架)中,这通常是通过在数据表中增加一个…...
使用LangGraph和LangSmith构建多智能体人工智能系统
现在,通过组合几个较小的子智能体来创建一个强大的人工智能智能体正成为一种趋势。但这也带来了一些挑战,比如减少幻觉、管理对话流程、在测试期间留意智能体的工作方式、允许人工介入以及评估其性能。你需要进行大量的反复试验。 在这篇博客〔原作者&a…...
【Linux】Linux 系统默认的目录及作用说明
博主介绍:✌全网粉丝23W,CSDN博客专家、Java领域优质创作者,掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域✌ 技术范围:SpringBoot、SpringCloud、Vue、SSM、HTML、Nodejs、Python、MySQL、PostgreSQL、大数据、物…...
MacOS下Homebrew国内镜像加速指南(2025最新国内镜像加速)
macos brew国内镜像加速方法 brew install 加速formula.jws.json下载慢加速 🍺 最新版brew安装慢到怀疑人生?别怕,教你轻松起飞! 最近Homebrew更新至最新版,每次执行 brew 命令时都会自动从官方地址 https://formulae.…...
【从零开始学习JVM | 第四篇】类加载器和双亲委派机制(高频面试题)
前言: 双亲委派机制对于面试这块来说非常重要,在实际开发中也是经常遇见需要打破双亲委派的需求,今天我们一起来探索一下什么是双亲委派机制,在此之前我们先介绍一下类的加载器。 目录 编辑 前言: 类加载器 1. …...
