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

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); 这条代码。

运行结果如下:

子进程没有受到处理变为僵尸进程,导致内存泄漏 

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

相关文章:

  • ctfshow刷题笔记—栈溢出—pwn61~pwn64
  • java23种设计模式-责任链模式
  • 新一代跨境电商ERP系统:从订单到发货的全流程自动化管理
  • 苹果廉价机型 iPhone 16e 影像系统深度解析
  • hive 面试题
  • VScode在windows10上使用clang-format
  • AWS API Gateway灰度验证实现
  • 【每日八股】MySQL篇(三):索引(上)
  • 在Pycharm中将ui文件修改为py文件
  • 看视频学习方法总结
  • Matlab 大量接单
  • 《深度剖析:生成对抗网络中生成器与判别器的高效协作之道》
  • Android6到Android15版本新增的功能和api
  • 【现代Web布局与动画技术:卡片组件实战分享】
  • 计算机网络之传输层(传输层提供的服务)
  • FPGA开发,使用Deepseek V3还是R1(1):应用场景
  • 哈希表和STL —— unorderde_set/unordered_map【复习笔记】
  • 计算机毕业设计SpringBoot+Vue.js体育馆使用预约平台(源码+文档+PPT+讲解)
  • 42 session反序列化漏洞
  • 【Jenkins】个人向-Jenkinsfile如何写
  • staruml绘制时序图和用例图
  • 问题修复-后端返给前端的时间展示错误
  • Rust配置开发环境+服务器实战
  • 使用DeepSeek+KIMI生成高质量PPT
  • 虚拟机如何设置ip
  • 蓝桥杯 路径之谜
  • Git操作指南:分支合并、回退及其他重要操作
  • Element Plus中el-tree点击的节点字体变色加粗
  • jenkens使用笔记
  • 腾讯混元文生图大模型(Hunyuan-DiT)与Stable Diffusion(SD)对比分析