Linux —— 信号(4)

Linux —— 信号(4)

  • 信号的处理
    • 用户态和内核态
  • 信号的捕捉
    • sigaction
  • sa_mask字段
  • volatile
  • SIGCHLD信号

我们今天接着来看信号

信号的处理

信号的处理简单一句话就是在内核态处理的。

用户态和内核态

用户态和内核态是操作系统和计组中的概念,我们这里提及一下:

用户态(User Mode)和内核态(Kernel Mode) 是操作系统的两种不同运行级别,它们在访问权限、可执行代码、以及运行环境上有所区别,以确保系统的稳定性和安全性。下面是用户态和内核态的主要区别:

  1. 访问权限
  • 内核态在内核态下,进程可以直接访问所有的系统资源,包括内存、I/O设备、系统核心数据等。这是因为内核态具有较高的权限,能够执行特权指令,比如修改内存管理单元(MMU)的设置、直接操作硬件等。
  • 用户态:相比之下,用户态下的进程权限较低,只能访问受限的资源,主要是自己的地址空间。用户态进程不能直接执行特权指令,也无法直接访问内核地址空间或硬件资源。
  1. 执行的代码
  • 内核态:当执行操作系统内核代码时,CPU处于内核态。这包括驱动程序、系统调用服务例程、中断和异常处理程序等。
  • 用户态:当执行用户程序的代码时,CPU处于用户态。大多数应用程序,如浏览器、文本编辑器等,都在用户态下运行。
  1. 切换方式
  • 用户态到内核态:切换通常发生在以下情况:系统调用(用户程序主动请求操作系统服务)、硬件中断(如键盘输入、网络数据到达)、异常(如除零错误、非法内存访问)。这些事件都会导致控制权从用户态转移到内核态,以便操作系统可以处理这些请求或事件。
  1. 安全性
  • 限制用户态程序的权限有助于保护系统稳定性,防止恶意或错误的用户程序破坏操作系统或其他用户的数据。内核态提供了必要的隔离和保护机制。
  1. 内存访问
  • 内核态可以访问整个内存空间,包括用户空间和内核空间;而用户态只能访问用户空间的内存,尝试访问内核空间的内存会触发硬件异常,进而可能导致进程被操作系统终止或产生其他错误响应。

通过这种区分,操作系统能够有效地管理资源、保护系统安全并提供稳定的服务环境。

我们不是学过进程地址空间吗?
在这里插入图片描述
我们知道,每一个进程都会有自己的进程地址空间:
在这里插入图片描述大家也看到了,我们的进程地址空间被划分成了两个部分,一个是用户空间,一个是内核空间

一般来说,我们写的东西都是在用户空间上,然后我们会有一张用户页表把我们进程地址空间上的东西映射到相应的物理内存上

在这里插入图片描述
同时,如果我们的代码要访问一些内核的东西,我们内核空间也有自己的页表来映射到相应的内存上
在这里插入图片描述

一般来说,内核的东西是不会变的,所以内核页表一般也只会有一张。

所以,如果我们要访问一些内核的东西,就要把自己切换为内核态然后通过内核页表去访问

那么对于信号来说是怎么处理的呢?

首先,我们在用户态发现了信号,会先切换为内核态:
在这里插入图片描述
如果我们自定义了信号的行为,会回到用户态:
在这里插入图片描述之后会重新回到内核态:
在这里插入图片描述

重新回到内核态会调用sigreturn
在这里插入图片描述
上面是一个大概的流程,如果搭建还不是很了解,可以看看这张图片:
在这里插入图片描述
用一张图表示的话,整个过程会有四次状态变化:
在这里插入图片描述

中间的交点可以进行信号捕捉:
在这里插入图片描述

信号的捕捉

sigaction

sigaction函数可以通过修改handle表,定义自己的handle行为:
在这里插入图片描述

sigaction是Unix/Linux系统中用于管理信号的一个函数,它是POSIX标准的一部分,提供了比传统signal函数更强大和灵活的信号处理机制。sigaction允许程序注册对特定信号的处理动作,以及配置与信号处理相关的额外选项,如信号掩码(暂时阻塞哪些信号)和是否重新设置信号处理函数为默认行为等。

基本用法如下:

#include <signal.h>int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact);

参数说明:

  • signum:要处理的信号的编号,例如SIGINT(对应Ctrl+C中断)。
  • act:指向一个struct sigaction结构体的指针,用于设置新的信号处理行为。这个结构体通常包含信号处理函数的指针(sa_handlersa_sigaction),一个信号掩码(sa_mask)表示在处理信号时应临时阻塞哪些其他信号,以及其他标志位。
  • oldact:(可选)指向另一个struct sigaction结构体的指针,用于保存之前对该信号的处理方式。如果对旧的行为不感兴趣,可以传入NULL

struct sigaction的定义大致如下:

struct sigaction {void (*sa_handler)(int);          // 信号处理函数指针,兼容旧式信号处理void (*sa_sigaction)(int, siginfo_t *, void *); // 新式信号处理函数指针,提供更多信号信息sigset_t sa_mask;                // 处理信号时应阻塞的信号集合int sa_flags;                    // 信号处理的标志,如SA_RESTART, SA_NODEFER等/* 其他可能的填充字段,取决于具体实现 */
};

使用sigaction而非signal的主要优势包括:

  • 更细粒度的控制,比如能够控制在处理信号期间哪些其他信号应该被阻塞。
  • 支持传递附加信息给信号处理函数,通过sa_sigactionsiginfo_t结构体。
  • 更可靠,因为它保证了信号处理函数的安装是原子操作,避免了race condition。
  • 可以设置信号处理函数是否被重新安装(SA_RESETHAND等标志)。

因此,sigaction函数常用于需要精确控制信号处理流程的高级编程中。

下面是以使用的例子:

#include <iostream>
#include <signal.h>
#include <unistd.h> // 添加头文件以使用getpid函数// 自定义信号处理函数
void handle(int signum) {std::cout << "get a sign: " << signum << std::endl;
}int main() {// 定义两个sigaction结构体变量,用于设置新的信号处理行为和保存原来的信号处理行为struct sigaction act, oact;// 配置act结构体,设置handle函数为信号2(SIGINT,默认为Ctrl+C)的处理函数act.sa_handler = handle;// 使用sigaction系统调用,将信号2的处理方式改为由handle函数处理// 同时,保存原先的信号处理方式到oact结构体中,尽管在这个示例中并未使用oactsigaction(SIGINT, &act, &oact);// 主循环,让进程持续运行并输出PIDwhile (1) {std::cout << "process is running, PID: " << getpid() << std::endl;// 让进程暂停1秒,避免无休止的输出占据终端sleep(1);}
}

sa_mask字段

这里要说明一下,如果我们正在处理某个信号,中间再次发送该信号,该信号会被屏蔽

#include <iostream>
#include <signal.h>
#include <unistd.h> // 添加头文件以使用getpid函数void PrintOpending(const sigset_t& opending);// 打印当前待处理信号集的函数
void PrintOpending(const sigset_t& opending) {for(int i = 1; i <= 31; ++i) { // 遍历常见的信号编号if(sigismember(&opending, i)) { // 检查该信号是否在待处理集合中std::cout << "1"; // 是,则输出1} else {std::cout << "0"; // 否,则输出0}}std::cout << std::endl; // 换行}// 自定义信号处理函数void handle(int signum) {sleep(1); // 等待一秒模拟信号处理时间std::cout << "catch a sign: " << signum << std::endl; // 输出接收到的信号编号while(true) { // 进入循环持续检查待处理信号sigset_t opending; // 创建一个信号集用于存放待处理的信号sigpending(&opending); // 获取当前进程的待处理信号集合PrintOpending(opending); // 打印当前待处理的信号状态sleep(1); // 每秒检查一次}}int main() {std::cout << "prcess is running PID: " << getpid() << std::endl; // 输出当前进程的PIDstruct sigaction act, oact; // 定义两个sigaction结构体变量// 配置act结构体,准备将handle函数设置为SIGINT信号的处理函数act.sa_handler = handle;// 使用sigaction系统调用,更改SIGINT信号的处理方式为handle函数// 同时,原SIGINT的处理方式被保存在oact中,但本例中并不使用这个信息sigaction(SIGINT, &act, &oact);// 主循环,让进程持续运行,但实际上由于没有具体执行内容,这里会一直占用CPUwhile (true) {}}

我们运行这段代码:
在这里插入图片描述
此时我们如果再按Ctrl + C:
在这里插入图片描述我们看到第二位已经变成了1,说明2号信号处于未决,说明2号信号已经被屏蔽了。

如果我们在处理2号信号时,不想让3号信号干扰,我们就要利用sa_mask添加另外的信号:

#include <iostream>
#include <signal.h>
#include <unistd.h> // 添加头文件以使用getpid函数void PrintOpending(const sigset_t& opending);// 打印当前待处理信号集的函数
void PrintOpending(const sigset_t& opending) {for(int i = 1; i <= 31; ++i) { // 遍历常见的信号编号if(sigismember(&opending, i)) { // 检查该信号是否在待处理集合中std::cout << "1"; // 是,则输出1} else {std::cout << "0"; // 否,则输出0}}std::cout << std::endl; // 换行
}// 自定义信号处理函数
void handle(int signum) {sleep(1); // 等待一秒模拟信号处理时间std::cout << "catch a sign: " << signum << std::endl; // 输出接收到的信号编号while(true) { // 进入循环持续检查待处理信号sigset_t opending; // 创建一个信号集用于存放待处理的信号sigpending(&opending); // 获取当前进程的待处理信号集合PrintOpending(opending); // 打印当前待处理的信号状态sleep(1); // 每秒检查一次}
}int main() {std::cout << "prcess is running PID: " << getpid() << std::endl; // 输出当前进程的PIDstruct sigaction act, oact; // 定义两个sigaction结构体变量// 配置act结构体,准备将handle函数设置为SIGINT信号的处理函数act.sa_handler = handle;// 使用sigaction系统调用,更改SIGINT信号的处理方式为handle函数// 同时,原SIGINT的处理方式被保存在oact中,但本例中并不使用这个信息sigemptyset(&act.sa_mask);sigaddset(&act.sa_mask,3); //同时将3号信号加入sigaction(SIGINT, &act, &oact);// 主循环,让进程持续运行,但实际上由于没有具体执行内容,这里会一直占用CPUwhile (true) {}}

在这里插入图片描述

此时,3号信号也被加入屏蔽集了。

不过,这种情况不是很常见,大家了解即可。

volatile

该关键字在C当中我们已经有所涉猎,今天我们站在信号的角度重新理解一下:

我们这里了解一下gcc,g++编译时候带的优化选项:
在使用g++编译C++代码时,可以通过添加优化选项来提高生成的可执行文件的执行效率。这些优化选项能够指导编译器以不同级别对代码进行优化,以减少程序的执行时间或占用的空间。以下是一些常用的g++编译时优化选项:

  1. -O1:进行基本的优化,提供了代码大小和执行速度之间的平衡。这是一个比较保守的优化级别,适合于调试和开发阶段。
  2. -O2:比-O1更进一步的优化,通常会提供更好的执行性能,可能会增加代码大小。这是推荐的优化等级,适用于大多数生产环境。
  3. -O3:这是最高的优化级别,提供了最积极的优化,可能会显著提升程序的运行速度,但也可能导致编译时间延长和代码体积增大。适合追求极致性能的应用。

使用示例:

g++ -O2 -DNDEBUG  main.cpp -o optimized_program

这条命令编译main.cpp,使用-O2进行优化,关闭调试信息(-DNDEBUG),最终生成名为optimized_program的可执行文件。

我们这里有这么一段代码:

#include <iostream>
#include <signal.h>
#include <unistd.h>
#include <stdio.h>int flag = 0;void handler(int sig)
{printf("chage flag 0 to 1\n");flag = 1;
}int main()
{signal(2, handler);while(!flag);printf("process quit normal\n");return 0;
}

如果我们正常编译,是没啥问题的:
在这里插入图片描述
但是如果带上-O2:
在这里插入图片描述
程序直接退出,因为编译器对flag做了优化,处理了死循环,如果我们不想让它优化,我们得使用volatile关键字:

#include <iostream>
#include <signal.h>
#include <unistd.h>
#include <stdio.h>volatile int flag = 0; //带上volatilevoid handler(int sig)
{printf("chage flag 0 to 1\n");flag = 1;
}int main()
{signal(2, handler);while(!flag);printf("process quit normal\n");return 0;
}

在这里插入图片描述
其实volatile是保证内存可见性

SIGCHLD信号

如果我们查看信号,会有这么一个信号:
在这里插入图片描述

SIGCHLD信号是在类Unix操作系统中,当一个子进程终止或者停止时,操作系统发送给其父进程的一种信号。这个信号的主要用途是通知父进程有关子进程状态的变化,以便父进程可以采取相应的行动,比如收集子进程的退出状态、资源清理等。以下是关于SIGCHLD信号的一些关键点:

  1. 默认行为:如果不特别设置,SIGCHLD信号的默认处理动作是忽略。这意味着父进程不会自动执行任何操作来响应子进程的终止,这可能导致子进程成为僵尸进程(zombie process)。
  2. 避免僵尸进程:父进程可以通过注册一个SIGCHLD信号处理函数,并在该函数中调用wait()waitpid()系统调用来回收子进程的状态信息,从而防止子进程变为僵尸进程。这样做可以让操作系统释放与子进程相关的资源。
  3. 自动重aping(Auto-reaping):如果父进程将SIGCHLD信号的处理设置为SIG_IGN(忽略),子进程在终止时会被内核自动清理,而不会生成僵尸进程。这种做法适用于那些不需要关注子进程具体退出状态的场景,例如某些高性能服务器。
  4. 非叠加性:SIGCHLD信号是不可累积的,也就是说,如果有多个子进程相继终止,父进程只会接收到一个SIGCHLD信号,而不是每个子进程一个。因此,在信号处理函数中可能需要使用循环调用wait()waitpid()来处理所有已终止的子进程。
  5. 信号处理策略:在编写多进程应用程序时,合理处理SIGCHLD信号非常重要,既可以避免资源泄露,又可以确保程序的健壮性。开发者可以根据应用的需求选择合适的处理方式,比如主动等待子进程结束、忽略信号或结合其他机制。
  6. 并发服务器中的应用:在并发服务器设计中,由于频繁创建和销毁子进程,正确处理SIGCHLD信号尤为重要,以防止系统中积累大量僵尸进程,影响系统性能。
#include<iostream>
#include<unistd.h>
#include<sys/wait.h>void handler(int signum)
{std::cout << "catch a sign: "<< signum << std::endl;
}int main()
{signal(17,handler);pid_t id = fork();if(id == 0){std::cout << "child is running Pid:" << getpid() << std::endl;sleep(10);exit(0);}int cnt = 5;while(cnt--){sleep(1);}wait(nullptr);
}

在这里插入图片描述

综上所述,SIGCHLD信号是管理子进程生命周期的关键机制,理解并正确处理它对于编写高效、稳定的多进程程序至关重要。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/689080.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Python快速入门-零基础也能掌握的编程技巧,基础方法和API整理

目录 前言 数据结构 数字 数学运算 随机数 字符串 列表 元组 字典 面向对象 JSON 文件操作 扩展 制作一个简易时钟 前言 环境什么就不在赘述&#xff0c;可以参考其他文章&#xff0c;也可以在线运行 CSDN在线运行地址&#xff1a;InsCode - 让你的灵感立刻落地…

优化资源利用,用C++内存池点亮编程之路

内存池介绍(Memory Pool): 它是一种内存分配方式&#xff0c;通过预先分配和复用内存块。 在真正使用内存之前&#xff0c;先申请一大块内存备用。当有新的内存需求时&#xff0c;就从内存池中分出一部分内存块&#xff0c; 若内存块不够再继续申请新的内存。如果我们不需要…

Linux学习笔记7---仿STM32自建寄存器库

为了开发方便&#xff0c;ST 官方为 STM32F103 编写了一个叫做 stm32f10x.h 的文件&#xff0c;在这个文件里面定义了 STM32F103 所有外设寄存器。而有些芯片是没有这种寄存器库的&#xff0c;在没有的情况下要学会自己建立一个寄存器库。NXP 官方并没有为 I.MX6UL 编写类似 st…

大模型微调之 在亚马逊AWS上实战LlaMA案例(十)

大模型微调之 在亚马逊AWS上实战LlaMA案例&#xff08;十&#xff09; 训练数据集格式 SageMaker JumpStart 目前支持域适应格式和指令调整格式的数据集。在本节中&#xff0c;我们指定两种格式的示例数据集。有关更多详细信息&#xff0c;请参阅附录中的数据集格式化部分。 …

嵌入式 - GPIO编程简介

An Introduction to GPIO Programming By Jeff Tranter Wednesday, June 12, 2019 编者按&#xff1a;本 2019 年博客系列是 ICS 最受欢迎的系列之一&#xff0c;现已更新&#xff08;2022 年 12 月&#xff09;&#xff0c;以确保内容仍然准确、相关和有用。 本博客是 Integr…

VBA_MF系列技术资料1-605

MF系列VBA技术资料1-605 为了让广大学员在VBA编程中有切实可行的思路及有效的提高自己的编程技巧&#xff0c;我参考大量的资料&#xff0c;并结合自己的经验总结了这份MF系列VBA技术综合资料&#xff0c;而且开放源码&#xff08;MF04除外&#xff09;&#xff0c;其中MF01-0…

机器学习-Numpy

机器学习-Numpy 如果一个人拒绝提高自己的思想觉悟&#xff0c;那么他只能处在弱小、可怜、凄惨的境地。 目录 机器学习-Numpy 1.Numpy&#xff1a;生成矩阵 做矩阵运算 1&#xff09;创建矩阵 ①使用列表创建 ②使用元组创建 2&#xff09;矩阵取值 3&#xff09;numpy…

Go实现树莓派读取at24c02 eeprom读写数据

步骤 启用i2c 参考 Go实现树莓派读取bh1750光照强度 代码 package mainimport ("fmt""periph.io/x/conn/v3/i2c" )type AT24C02Device struct {dev *i2c.Dev }func NewAT24C02Device(addr uint16, bus i2c.BusCloser) (*AT24C02Device, error) {var (d…

Springboot+Vue项目-基于Java+MySQL的影院订票系统(附源码+演示视频+LW)

大家好&#xff01;我是程序猿老A&#xff0c;感谢您阅读本文&#xff0c;欢迎一键三连哦。 &#x1f49e;当前专栏&#xff1a;Java毕业设计 精彩专栏推荐&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb; &#x1f380; Python毕业设计 &…

Linux——基础IO(1)

前言 铺垫&#xff1a;文件 1.之前我们讲过文件内容属性 磁盘中创建一个空文件也要占空间(就算内容为空&#xff0c;文件属性也占空间) 文件操作文件内容的操作文件属性的操作 有可能在操作文件的过程中既改变内容又改变属性 2.访问文件之前&#xff0c;都得先打开文件 修改文…

HTTP基础概念和HTTP缓存技术

什么是HTTP HTTP是超文本传输协议&#xff0c;主要分为三个部分&#xff1a;超文本、传输、协议。 超文本是指&#xff1a;文字、图片、视频的混合体。传输是指&#xff1a;点与点之间的信息通信。协议是指&#xff1a;通信时的行为规范或约定 HTTP常见字段 字段名 解释 例…

2024年人工智能数据报告

大家好&#xff0c;我是爱编程的喵喵。双985硕士毕业&#xff0c;现担任全栈工程师一职&#xff0c;热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。现为CSDN博客专家、人工智能领域优质创作者。…