[linux]进程信号(信号的概念,信号的产生方式,信号的相关接口、指令,函数,信号怎么保存(原理),信号怎么处理)

目录

一、信号的概念

二、信号的产生方式

通过键盘发送信号

通过系统调用,指令

异常

软件条件

三、信号怎么保存(原理)

信号其他相关常见概念

在内核中表示 

sigset_t 

四、信号的相关接口、指令,函数 

signal

sigprocmask

sigpending

sigemptyset

sigfillset

sigaddset

sigdelset

sigismember

sigaction 

五、信号怎么处理

信号什么时候被处理?

信号的捕捉图


一、信号的概念

信号是进程之间事件异步通知的一种方式,属于软中断。

在linux中通过指令:kill -l 可以查看信号列表

二、信号的产生方式

通过键盘发送信号

如常用的ctrl + c就是向前台进程发送2号信号(SIGINT)

通过系统调用,指令

如:

kill -2 1234,意思是向pid为1234的进程发送2号信号

或者

#include <sys/types.h>
#include <signal.h>

int kill(pid_t pid, int sig);

作用:向指定pid的进程发送sig信号

返回值:如果成功(至少发送了一个信号),则返回0。如果出现错误,则返回-1,并适当地设置errno

异常

硬件异常被硬件以某种方式被硬件检测到并通知内核,然后内核向当前进程发送适当的信号。例如当前进程执行了除以0的指令,CPU的运算单元会产生异常,内核将这个异常解释为SIGFPE(8号)信号发送给进程。再比如当前进程访问了非法内存地址,,MMU会产生异常,内核将这个异常解释为SIGSEGV(11号)信号发送给进程。

软件条件

例如:

#include <unistd.h>
unsigned int alarm(unsigned int seconds);
调用alarm函数可以设定一个闹钟(闹钟也是一种软件条件),也就是告诉内核在seconds秒之后给当前进程发SIGALRM(14号)信号, 该信号的默认处理动作是终止当前进程。这个函数的返回值是0或者是以前设定的闹钟时间还余下的秒数。

三、信号怎么保存(原理)

信号其他相关常见概念

  • 实际执行信号的处理动作称为信号递达(Delivery)

  • 信号从产生到递达之间的状态,称为信号未决(Pending)。

  • 进程可以选择阻塞 (Block )某个信号。

  • 被阻塞的信号产生时将保持在未决状态,直到进程解除对此信号的阻塞,才执行递达的动作。

  • 注意,阻塞和忽略是不同的,只要信号被阻塞就不会递达,而忽略是在递达之后可选的一种处理动作。

在内核中表示 

  • 每个信号都有两个标志位分别表示阻塞(block)和未决(pending),还有一个函数指针表示处理动作。信号产生时,内核在进程控制块中设置该信号的未决标志,直到信号递达才清除该标志。在上图的例子中,SIGHUP信号未阻塞也未产生过,当它递达时执行默认处理动作。
  • SIGINT信号产生过,但正在被阻塞,所以暂时不能递达。虽然它的处理动作是忽略,但在没有解除阻塞之前不能忽略这个信号,因为进程仍有机会改变处理动作之后再解除阻塞。
  • SIGQUIT信号未产生过,一旦产生SIGQUIT信号将被阻塞,它的处理动作是用户自定义函数sighandler。如果在进程解除对某信号的阻塞之前这种信号产生过多次,将如何处理?POSIX.1允许系统递送该信号一次或多次。Linux是这样实现的:常规信号在递达之前产生多次只计一次,而实时信号在递达之前产生多次可以依次放在一个队列里。

sigset_t 

从上图来看,每个信号只有一个bit的未决标志,非0即1,不记录该信号产生了多少次,阻塞标志也是这样表示的。因此,未决和阻塞标志可以用相同的数据类型sigset_t来存储,sigset_t称为信号集,这个类型可以表示每个信号的“有效”或“无效”状态,在阻塞信号集中“有效”和“无效”的含义是该信号是否被阻塞,而在未决信号集中“有效”和“无效”的含义是该信号是否处于未决状态。

四、信号的相关接口、指令,函数 

signal

#include <signal.h>

typedef void (*sighandler_t)(int);

sighandler_t signal(int signum, sighandler_t handler);

功能:对信号进行自定义捕捉

参数:

signum:信号,如2号信号就填2或者SIGINT,注意:SIGKILL(9号)0和SIGSTOP(19号)不可忽略和自定义捕捉                                                           

handler:一个函数指针,函数内容为自定义的操作

返回值:成功返回自定义的函数指针,失败返回SIG_ERR。

 使用例子:

#include <signal.h>
#include <sys/types.h>
#include <unistd.h>

#include <iostream>
using namespace std;
void handler(int signo)
{
    while (1)
    {
        cout << "我是" << signo << "号信号,我被自定义捕捉了" << endl;
        sleep(1);

    }
}
int main()
{
    cout << getpid() << endl;
    signal(2, handler);
    while (1)
    {
        cout << "我没有被自定义捕捉" << endl;
        sleep(1);
    }
    return 0;
}

sigprocmask

 #include <signal.h>

 int sigprocmask(int how, const sigset_t *set, sigset_t *oldset);
功能:读取或更改进程的信号屏蔽字(阻塞信号集,其存储结构为位图)

参数:

how:读取或更改方式

有三种选择:

假设当前的信号屏蔽字为mask。

  1. SIG_BLOCK:set包含了我们希望添加到当前信号屏蔽字的信号,相当于mask=mask|set
  2. SIG_UNBLOCK:set包含了我们希望从当前信号屏蔽字中解除阻塞的信号,相当于mask=mask&~set
  3. SIG_SETMASK:设置当前信号屏蔽字为set所指向的值,相当于mask=set

set:如果set不为非空指针,则按照set更改进程的屏蔽字

oldset:如果set和oldset都不为非空指针,则把先前信号的屏蔽字备份给oldset,再更改进程的屏蔽字,如果不许要备份先前信号的屏蔽字,则此参数设为nullptr

sigpending

  #include <signal.h>

  int sigpending(sigset_t *set);

功能:读取当前进程的未决信号集,通过set参数传出。调用成功则返回0,出错则返回-1。

sigemptyset

sigfillset

sigaddset

sigdelset

sigismember

#include <signal.h>

int sigemptyset(sigset_t *set);

int sigfillset(sigset_t *set);

int sigaddset(sigset_t *set, int signum);

int sigdelset(sigset_t *set, int signum);

int sigismember(const sigset_t *set, int signum);

功能:

sigemptyset()将set给出的信号集初始化为空,将所有信号从该集合中排除。

sigfillset()初始化set为full(比特位全为1),包括所有信号。

sigaddset()和sigdelset()分别从set中添加和删除信号符号。

sigismember()测试sigum是否是set的成员。

set:阻塞信号集

返回值:

sigemptyset()、sigfillset()、sigaddset()和sigdelset()成功时返回0,错误时返回-1。

sigismember()如果signum是set的成员返回1,如果signum不是set的成员返回0,如果出错返回-1。当出现错误时,这些函数设置errno来指示原因。

 上面函数的综合使用例子

#include <signal.h>
#include <sys/types.h>
#include <unistd.h>
#include <iostream>
using namespace std;

void printsigset(sigset_t *s)
{
    for (int i = 31; i > 0; i--)
    {
        if (sigismember(s, i))
        {
            cout << "1";
        }
        else
        {
            cout << "0";
        }
    }
    cout << endl;
}
int main()
{
    sigset_t s, p, o;
    sigemptyset(&s);
    for (int i = 1; i < 32; i++)
    {
        sigaddset(&s, i);
        sigprocmask(SIG_BLOCK, &s, &p);
        cout << "前" << ": ";
        printsigset(&p);
        cout << "后" << ": ";
        printsigset(&s);
       sleep(2);
    }
    return 0;
}

解释代码

结果:1到31号信号都被阻塞(实际上9号和19号信号无法阻塞),以位图的形式打印出来

sigaction 

#include <signal.h>

int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact);

解释:

sigaction函数可以读取和修改与指定信号相关联的处理动作。调用成功则返回0,  出错则返回- 1。signum是指定信号的编号。若act指针非空,则根据act修改该信号的处理动作。若oldact指针非空,则通过oldact传出该信号原来的处理动作。act和oldact指向sigaction结构体。

sigaction结构体:

struct sigaction

{

    void (*sa_handler)(int);

    void (*sa_sigaction)(int, siginfo_t *, void *);

    sigset_t sa_mask;

    int sa_flags;

    void (*sa_restorer)(void);

};

例子:

 使用sigaction函数自定义SIGINT信号的处理方式。要求:自定义处理函数名称为“sigcb”, 在sigcb当中完成打印信号值。

#include <signal.h>
#include <unistd.h>
#include <iostream>
using namespace std;

void sigcb(int signo)
{
    while (1)
    {
        cout << "我是" << signo << "号信号, 我被自定义处理了" << endl;
        sleep(2);
    }
}
int main()
{
    struct sigaction act;
    act.sa_handler = sigcb;
    sigaction(SIGINT, &act, NULL);

    while (1)
    {
        cout << "我没有被自定义捕捉" << endl;
        sleep(2);
    }
}

 结果:

五、信号怎么处理

信号什么时候被处理?

信号发出后并不是立马递达的,而是进程从内核态返回用户态的时候,才对信号进行检查与处理。

信号的捕捉图

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

数据结构--树的遍历

数据结构--树的遍历 1. 前序中序后序遍历2. 前序中序后序遍历代码 1. 前序中序后序遍历 2. 前序中序后序遍历代码 /** public class TreeNode {int val 0;TreeNode left null;TreeNode right null;public TreeNode(int val) {this.val val;}} */// 前序遍历顺序&#xff1…

SQL函数学习记录

聚合函数 函数是编程语言的基础之一&#xff0c;在对数字的运算中&#xff0c;我们用的最多的就是聚合函数&#xff0c;本篇接下来就详细阐述下SQL中聚合函数的运用。 什么是聚合函数&#xff08;aggregate function&#xff09;&#xff1f; 聚合函数指的是对一组值执行计算…

MYSQL05高级_查看修改存储引擎、InnoDB和MyISAM对比、其他存储引擎介绍

文章目录 ①. 查看、修改存储引擎②. InnoDB和MyISAM对比③. Archive引擎 - 归档④. Blackhole引擎丢数据⑤. CSV - 引擎⑥. Memory引擎 - 内存表⑦. Federated引擎 - 访问远程表⑧. Merge引擎 - 管理多个MyISAM⑨. NDB引擎 - 集群专用 ①. 查看、修改存储引擎 ①. 查看mysql提…

C++ 之LeetCode刷题记录(三十六)

&#x1f604;&#x1f60a;&#x1f606;&#x1f603;&#x1f604;&#x1f60a;&#x1f606;&#x1f603; 开始cpp刷题之旅。 目标&#xff1a;执行用时击败90%以上使用 C 的用户。 16. 最接近的三数之和 给你一个长度为 n 的整数数组 nums 和 一个目标值 target。请你…

备战蓝桥杯Day20 - 堆排序的实现

一、每日一题 蓝桥杯真题之互质数的个数 我的解法&#xff1a; 两个数互质&#xff0c;说明两个数的最大公约数是1&#xff0c;定义了一个函数判断两个数的最大公约数&#xff0c;再在循环中判断&#xff0c;并实现计数。可以实现运行&#xff0c;缺点是时间复杂度很高&#…

【Leetcode每日一题】二分查找 - 寻找旋转排序数组中的最小值(难度⭐⭐)(22)

1. 题目解析 Leetcode链接&#xff1a;153. 寻找旋转排序数组中的最小值 这个题目乍一看很长很复杂&#xff0c;又是旋转数组又是最小值的 但是仔细想想&#xff0c;结合题目给的示例&#xff0c;不难看出可以用二分的方法来解决 核心在于找到给定数组里面的最小值 2. 算法原…

预训练大模型LLM的PEFT之—— Prefix Tuning

简介 Prefix Tuning是2021.01提出来的&#xff0c;在它之前&#xff0c;我们使用prompt主要是人工设计模板或者自动化搜索模板&#xff0c;也就是prompt范式的第一阶段&#xff0c;就是在输入上加上prompt文本&#xff0c;再对输出进行映射。这种离散模板对模型的鲁棒性很差。…

如何在Window系统部署BUG管理软件并结合内网穿透实现远程管理本地BUG

文章目录 前言1. 本地安装配置BUG管理系统2. 内网穿透2.1 安装cpolar内网穿透2.2 创建隧道映射本地服务3. 测试公网远程访问4. 配置固定二级子域名4.1 保留一个二级子域名5.1 配置二级子域名6. 使用固定二级子域名远程 前言 BUG管理软件,作为软件测试工程师的必备工具之一。在…

MYSQL学习笔记:索引

MYSQL学习笔记&#xff1a;索引 文章目录 MYSQL学习笔记&#xff1a;索引索引的分类索引的创建删除索引优化B树索引B树InnoDB主键和二级索引树聚集索引与非聚集索引哈希索引INNODB的自适应哈希索引索引和慢查询 用索引也是要涉及磁盘I/O的操作的索引也是一种数据结构&#xff0…

c语言day4 运算符 表达式 三大控制结构

1&#xff1a; 2&#xff1a; 输入一个年月日 计算这是这一年的第几天 17 int year,month,day;18 printf("请输入年份 月份 日期");19 scanf("%d %d %d",&year,&month,&day);20 int feb28;21 if(year%40&&year%1…

哈工大中文mistral介绍(Chinese-Mixtral-8x7B)

Chinese-Mixtral-8x7B基于Mistral发布的模型Mixtral-8x7B进行了中文扩词表增量预训练。扩充后的词表显著提高了模型对中文的编解码效率&#xff0c;并通过大规模开源语料对扩词表模型进行增量预训练&#xff0c;使模型具备了强大的中文生成和理解能力。 开源地址见https://gith…

前端-DOM树

dom树描述网页元素关系的一个专有名词&#xff0c;如html内包含了head、body&#xff0c;而head内包含meta、title、script等&#xff0c;body内包含div等元素&#xff1b;网页所有内容都在document里面&#xff0c;网页内容以树状形式排列&#xff0c;所以称之为dom树 dom树内…

centos服务配置springboot服务开机启动

在做后端服务运维时&#xff0c;经常遇到服务器重启时&#xff0c;需要移动一堆后端服务。服务器故障自动重启时&#xff0c;通常无人通知。把springboot服务的jar包配置开机启动太有必要了&#xff0c;虽然不是很复杂&#xff0c;这里记录一下太有必要了。 创建jar包启动和停…

看完这篇,终于理解如何制作产品使用说明书啦!

制作产品说明书是一项重要的任务&#xff0c;它不仅提供了产品的详细信息&#xff0c;还可以帮助用户正确地使用和维护产品&#xff0c;确保产品说明书的质量和可使用性。在制作产品说明书时&#xff0c;掌握注意事项&#xff0c;可以帮助你更加高效地制作产品说明书。以下是Lo…

lv20 QT主窗口

熟悉创建主窗口项目 1 QAction 2 主窗口 菜单栏&#xff1a;fileMenu menuBar()->addMenu(tr("&File")); 工具栏&#xff1a;fileToolBar addToolBar(tr("File")); 浮动窗&#xff1a;QDockWidget *dockWidget new QDockWidget(tr("Dock W…

基于Python3的数据结构与算法 - 06 topk问题

一、引入 问题&#xff1a;目前共有n个数&#xff0c;设计算法得到前k大的数。&#xff08;m<n&#xff09; 解决思路&#xff1a; 排序后切片&#xff1a;O(n*lognm) O(n*logn)排序LowB三人组&#xff1a;O(mn) 例如冒泡排序&#xff0c;交换m次&#xff0c;即可取前m…

uniapp 安装安卓、IOS模拟器并调试

一、安装Android模拟器并调试 1.下载并安装Android Studio。 2.创建简单project。 3.安装模拟器。 完成安卓模拟器的安装。 4.启动模拟器。 5.hbuilderx选择模拟器、运行。 点击刷新按钮后出现模拟器&#xff0c;勾选并运行。 6.调试。 在 HBuilderX 中&#xff0c;项目启…

如何将字体添加到 ONLYOFFICE 桌面编辑器8.0

作者&#xff1a;VincentYoung 为你写好的文字挑选一款好看的字体然而自带的字体列表却找不到你喜欢的怎么办&#xff1f;这只需要自己手动安装一款字体即可。这里教你在不同的桌面操作系统里的多种字体安装方法。 ONLYOFFICE 桌面编辑器 ONLYOFFICE 桌面编辑器是一款免费的办…

Java设计模式 | 七大原则之合成复用原则

基本介绍 合成复用原则&#xff08;Composite Reuse Principle&#xff09;尽量使用合成/聚合的方式&#xff0c;而不是使用继承 设计原则核心思想总结 找出应用中可能需要变化之处&#xff0c;把他们独立出来&#xff0c;不要和那些不需要变化的代码混在一起针对接口编程&…

HQYJ 2024-2-29 作业

作业&#xff1a; 编写链表&#xff0c;链表里面随便搞点数据 使用 fprintf 将链表中所有的数据&#xff0c;保存到文件中 使用 fscanf 读取文件中的数据&#xff0c;写入链表中 #include <stdio.h> #include <string.h> #include <stdlib.h> #include <…