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

Linux信号一门搞定

1.信号是什么?

信号其实就是一个软件中断。

例:

  1. 输入命令,在Shell下启动一个前台进程。
  2. 用户按下Ctrl-C,键盘输入产生一个硬件中断。
  3. 如果CPU当前正在执行这个进程的代码,则该进程的用户空间代码暂停执行, CPU从用户态切换到内核态处理硬件中断。
  4. 终端驱动程序将Ctrl-C解释成一个SIGINT信号,记在该进程的PCB中(也可以说发送了一个SIGINT信号给该进程)。
  5. 当某个时刻要从内核返回到该进程的用户空间代码继续执行之前,首先处理PCB中记录的信号,发现有一个SIGINT信号待处理,而这个信号的默认处理动作是终止进程,所以直接终止进程而不再返回它的用户空间代码执行。

在这个例子中,由ctrl+c产生的硬件中断就是一个信号。Ctrl+C产生的信号只能发送给前台进程,命令后加&就可放到后台运行。
Shell可同时运行一个前台进程和任意多个后台进程,只有前台进程才能接受到像CTRL+C这种控制键产生的信号。

2.信号的种类

使用命令查看:

kill -l

非可靠信号:1~31号信号,信号可能会丢失
可靠信号:34~64号信号,信号不可能丢失

SIGHUP:1号信号,Hangup detected on controlling terminal or death of controlling process(在控制终端上挂起信号,或让进程结束),ation:term

SIGINT:2号信号,Interrupt from keyboard(键盘输入中断,ctrl + c ),action:term

SIGQUIT:3号信号,Quit from keyboard(键盘输入退出**,ctrl+ |** ),action:core,产生core dump文件

SIGABRT:6号信号,Abort signal from abort(3)(非正常终止,double free),action:core

SIGKILL:9号信号,Kill signal(杀死进程信号),action:term,该信号不能被阻塞、忽略、自定义处理

SIGSEGV:11号信号,Invalid memory reference(无效的内存引用,解引用空指针、内存越界访问),action:core

SIGPIPE:13号信号,Broken pipe: write to pipe with no readers(管道中止: 写入无人读取的管道,会导致管道破裂),action:term

SIGCHLD:17号信号,Child stopped or terminated(子进程发送给父进程的信号,但该信号为忽略处理的)

SIGSTOP:19号信号,Stop process(停止进程),action:stop

SIGTSTP:20号信号,Stop typed at terminal(终端上发出的停止信号,ctrl + z),action:stop

具体的信号采取的动作和详细信息可查看:man 7 signal

3.信号的产生

3.1硬件产生

硬件产生即通过终端按键产生的信号:

  1. ctrl + c:SIGINT(2),发送给前台进程,& 进程放到后台运行,fg 把刚刚放到后台的进程,再放到前台来运行
  2. ctrl + z:SIGTSTP(20),一般不用,除非有特定场景
  3. ctrl + | :SIGQUIT(3),产生core dump文件

产生core dump文件的条件:

当前OS一定不要限制core dump文件的大小,ulimit -a
磁盘空间要足够
如何产生:
3.1 解引用空指针,收到11号信号,产生core dump文件
3.2 内存访问越界,程序一旦崩溃,就会收到11号信号,也就会产生core dump文件
3.3 double free,收到6号信号,并产生core dump。
3.4 free(NULL),不会崩溃

3.2软件产生

软件产生即调用系统函数向进程发信号

  1. kill函数
#include <sys/types.h>
#include <signal.h>
int kill(pid_t pid, int sig);
参数解释:
pid:进程号
sig:要发送的信号值
返回值:成功返回0,失败返回-1,并设置错误
  1. kill命令:kill -[信号] pid,
  2. abort:void abort(void);,收到6号信号,谁调用该函数,谁就收到信号
  3. alarm:unsigned int alarm(unsigned int seconds);,收到14号信号,告诉内核在seconds秒后给进程发送SIGALRM信号,该信号默认处理动作为终止当前进程。

4.信号的注册

信号注册又分为可靠信号的注册和非可靠信号的注册。
信号注册实际上是一个位图和一个sigqueue队列。

4.1非可靠信号的注册

当进程收到非可靠信号时:

  1. 将非可靠信号对应的比特位置为1
  2. 添加sigqueue节点到sigqueue队列当中,但是,在添加sigqueue节点的时候,队列当中已然有了该信号的sigqueue节点,则不添加

4.2可靠信号的注册

当进程所受到可靠信号时:

在sig位图中更改信号对应的比特位为1
不论之前sigqueue队列中是否存在该信号的sigqueue节点,都再次添加sigqueue节点到sigqueue队列当中去

5.信号的注销

5.1非可靠信号的注销

信号对应的比特位从1置为0
将该信号的sigqueue节点从sigqueue队列当中进行出队操作

5.2可靠信号的注销

将该信号的sigqueue节点从sigqueue队列当中进行出队操作
需要判断sigqueue队列当中是否还有相同的sigqueue节点:
①没有了:信号比特位从1置为0
②还有:不会更改sig位图中的比特位

6.信号阻塞

6.1信号是怎样阻塞的?

信号的阻塞,并不会干扰信号的注册。信号能注册,但不能被立即处理,
将block位图中对应的信号比特位置为1,表示阻塞该信号
进程收到该信号,还是一如既往的注册
当进程进入到内核空间,准备返回用户空间的时候,调用do_signal函数,就不会立即去处理该信号了
当该信号不被阻塞后,就可以进行处理了

6.2sigprocmask

函数原型:int sigprocmask(int how, const sigset_t *set, sigset_t *oldset);
参数解释:

how,该做什么样的操作
SIG_BLOCK:设置信号为阻塞
SIG_UNBLOCK:解除信号阻塞
SIG_SETMASK:替换阻塞位图
set:用来设置阻塞位图
SIG_BLOCK:设置某个信号为阻塞,block(new) = block(old) | set
SIG_UNBLOCK:解除某个信号阻塞,block(new)= block(old) &~set)
SIG_SETMASK:替换阻塞位图,block(new)= set
oldset:原来的阻塞位图

例:下述例子,信号全部被阻塞,采用kill -9,将该进程结束掉

#include <stdio.h>
#include <signal.h>
#include <unistd.h>void signcallback(int signumber)
{printf("change the signal %d\n",signumber);
}int main()
{sigset_t set;sigset_t oldset;sigfillset(&set);//所有比特位全置为1,则信号全部会被阻塞sigprocmask(SIG_BLOCK,&set,&oldset);while(1){sleep(1);}return 0;
}

结果: 此时发送信号是不会有作用的,采用kill -9强杀掉

7.信号未决

7.1 未决概念

实际执行信号的处理动作称为信号递达(Delivery),信号从产生到递达之间的状态,称为信号未决(Pending)。
进程可以选择阻塞(Block)某个信号。被阻塞的信号产生时将保持在未决状态,直到进程解除对此信号的阻塞,才执行递达的动作。注意,阻塞和忽略是不同的,只要信号被阻塞就不会递达,而忽略是、在递达之后可选的一种处理动作。

7.2 sigpending

函数原型:int sigpending(sigset_t *set);
读取当前进程的未决信号集,通过set参数传出。调用成功返回0,出错返回-1.

例:

#include <stdio.h>
#include <unistd.h>
#include <signal.h>void signalcallback(int signumber)
{printf("chang signumber %d\n",signumber);
}
void printsigset(sigset_t *set)
{int i = 0;for(;i < 32;i++){if(sigismember(set,i)){putchar('1');}else{putchar('0');}}
}int main()
{signal(2,signalcallback);signal(10,signalcallback);sigset_t set;sigset_t oldset;sigset_t pending;sigfillset(&set);//所有比特位全部置为1,则信号会全部被阻塞sigprocmask(SIG_BLOCK,&set,&oldset);while(1){sigpending(&pending);printsigset(&pending);sleep(1);}return 0;
}

结果:

8.信号的处理方式

每个信号都有两个标志位分别表示阻塞和未决,还有一个函数指针表示处理动作。

在上述例子中:

  1. SIGHUP信号未阻塞也未产生过,当它递达时执行默认处理动作。
  2. SIGINT信号产生过,但正在被阻塞,所以暂时不能递达。虽然它的处理动作是忽略,但在没有解除阻塞之前不能忽略这个信号,因为进程仍有机会改变处理动作之后再解除阻塞。
  3. SIGQUIT信号未产生过,一旦产生SIGQUIT信号将被阻塞,它的处理动作是用户自定义函数sighandler。

8.1signal函数

该函数可以更改信号的处理动作。

typedef void (*sighandler_t)(int);
sighandler_t signal(int signum, sighandler_t handler);
参数解释:signum:更改的信号值
handler:函数指针,要更改的动作是什么

实际上,该函数内部也调用了sigaction函数。

8.2sigaction函数

读取和修改与指定信号相关联的处理动作。

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

参数解释:

signum:待更改的信号值

struct sigaction结构体:

void     (*sa_handler)(int);//函数指针,保存了内核对信号的处理方式
void     (*sa_sigaction)(int, siginfo_t *, void *);//
sigset_t   sa_mask;//保存的是当进程在处理信号的时候,收到的信号
int        sa_flags;//SA_SIGINFO,OS在处理信号的时候,调用的就是sa_sigaction函数指针当中
//保存的值0,在处理信号的时候,调用sa_handler保存的函数
void     (*sa_restorer)(void);

例:

#include <stdio.h>
#include <unistd.h>
#include <signal.h>void signcallback(int signumber)
{printf("change signumber %d\n",signumber);
}int main()
{struct sigaction act;//act为入参sigemptyset(&act.sa_mask);act.sa_flags = 0;act.sa_handler = signcallback;struct sigaction oldact;//oldact为出参sigaction(3,&act,&oldact);while(1){sleep(1);}return 0;
}

结果:

8.3 自定义信号处理的流程

  1. task_struct结构体中有一个struct sighand_struct结构体。
  2. struct sighand_struct结构体有一个**struct k_sigaction action[_NSIG]**结构体数组。
  3. 该数组中,其中的**_sighandler_t sa_handler**保存的是信号的处理方式,通过改变其指向,可以实现我们对自定义信号的处理。

9.信号的捕捉

9.1信号捕捉的条件

如果信号的处理动作是用户自定义函数,在信号递达时就调用这个函数,这就称为信号捕捉。

9.2信号捕捉流程

内核态返回用户态会调用do_signal函数,两种情况:

  1. 无信号:sys_return函数,返回用户态

  2. 有信号:先处理信号,信号返回,再调用do_signal函数
    例:

  3. 程序注册了SIGQUIT信号的处理函数sighandler。

  4. 当前正在执行main函数,这时发生中断或异常切换到内核态。

  5. 在中断处理完毕后要返回用户态的main函数之前检查到有信号SIGQUIT递达。

  6. 内核决定返回用户态后不是恢复main函数的上下文继续执行,而是执行sighandler函数, sighandler和main函数使用不同的堆栈空间,它们之间不存在调用和被调用的关系,是两个独立的控制流程。

  7. sighandler函数返回后自动执行特殊的系统调用sigreturn再次进入内核态。

  8. 如果没有新的信号要递达,这次再返回用户态就是恢复main函数的上下文继续执行了。

10.常用信号集操作函数

int sigemptyset(sigset_t *set);//将比特位图全置为0int sigfillset(sigset_t *set);//将比特位图全置为1int sigaddset(sigset_t *set, int signum);//将该set位图,多少号信号置为1int sigdelset(sigset_t *set, int signum);//将该set位图,多少号信号置为0int sigismember(const sigset_t *set, int signum);//信号signum是否是set位图中的信号

11.SIGCHLD信号

该信号是子进程在结束是发送给父进程的信号,但是该信号的处理方式是默认处理的。
父进程对子进程发送过来的SIGCHLD信号进行了忽略处理,就会导致子进程成为僵尸进程。

可以自定义该信号的处理方式:

#include <stdio.h>
#include <unistd.h>
#include <signal.h>
#include <string.h>
#include <sys/wait.h>
#include <stdlib.h>void signcallback(int signumber)
{printf("change signal %d\n",signumber);wait(NULL);
}int main()
{signal(17,signcallback);pid_t pid = fork();if(pid < 0){perror("fork");return -1;}else if(pid == 0){printf("I am child\n");sleep(1);exit(12);}else{while(1){sleep(1);}}return 0;
}

指令查看后台:ps aux | grep ./fork

原文地址:
https://blog.csdn.net/w903414/article/details/109802539?utm_source=app&app_version=4.18.0&utm_source=app

相关文章:

Linux信号一门搞定

1.信号是什么&#xff1f; 信号其实就是一个软件中断。 例&#xff1a; 输入命令&#xff0c;在Shell下启动一个前台进程。用户按下Ctrl-C&#xff0c;键盘输入产生一个硬件中断。如果CPU当前正在执行这个进程的代码&#xff0c;则该进程的用户空间代码暂停执行&#xff0c;…...

手撸一个动态Feign,实现一个“万能”接口调用

Feign&#xff0c;在微服务框架中&#xff0c;是的服务直接的调用变得很简洁、简单&#xff0c;而不需要再编写Java Http调用其他微服务的接口。 动态feign 对于fegin调用&#xff0c;我们一般的用法&#xff1a;为每个微服务都创建对应的feignclient接口&#xff0c;然后为每…...

Linux Capabilities 入门

目录 Linux capabilities 是什么&#xff1f; capabilities 的赋予和继承 线程的 capabilities Permitted Effective Inheritable Bounding Ambient 文件的 capabilities Permitted Inheritable Effective 运行 execve() 后 capabilities 的变化 案例 Linux capab…...

驱动 day6

关于设备树的理解&#xff1a; 设备树&#xff08;Device Tree&#xff09;是一种用于特定硬件设备的解释语法树。它用来表示存储有关主板硬件和CPU架构信息的数据在内核中的传递格式&#xff0c;使内核可以更好地了解硬件并支持它们&#xff0c;而不必编写固定的代码。设备节点…...

附录2-tensorflow目标检测

源码来自作者Bubbliiiing&#xff0c;我对参考链接的代码略有修改&#xff0c;网盘地址 链接&#xff1a;百度网盘 请输入提取码 提取码&#xff1a;dvb1 目录 1 参考链接 2 环境 3 数据集准备 3.1 VOCdevkit/VOC2007 3.2 model_data/voc_classes.txt 3.3 voc_an…...

常见的EMC问题

电磁兼容设计的目的就在于满足产品功能要求、减少调试时间&#xff0c;使产品满足电磁兼容标准的要求&#xff0c;并且使产品不会对系统中的其它设备产生电磁干扰。 电磁兼容设计中常见的问题有哪些&#xff1f; 1、电磁兼容设计可以从电路设计&#xff08;包括器件选择&…...

Redis内存存储效率问题

目录 内存碎片是如何形成的&#xff1f; 如何判断是否有内存碎片&#xff1f; 如何清理内存碎片&#xff1f; INFO命令 面向 Prometheus 的 Redis-exporter 监控 实习期间&#xff0c;了解到&#xff0c;企业级开发中多个项目使用Redis&#xff0c;运行Redis实例的有可能是…...

3.28 haas506 2.0开发教程-example-蓝牙多设备扫描(仅支持M320,HD1)

haas506 2.0开发教程-example-蓝牙多设备扫描案例说明蓝牙信息克隆1.手机蓝牙改名信息克隆代码测试案例说明 开发板扫描蓝牙设备&#xff0c;获取并打印蓝牙设备mac地址。mac地址每个设备不同&#xff0c;且不能更改。本案例仅适用于M320开发板和HD1-RTU。案例使用手机与iBeac…...

C语言经典编程题100例(41~60)

目录41、习题4-4 特殊a串数列求和42、习题4-6 水仙花数43、习题4-7 最大公约数和最小公倍数44、习题7-5 找鞍点45、练习5-1 求m到n之和46、练习5-2 找两个数中最大者47、练习5-3 数字金字塔48、习题5-1 符号函数49、习题5-2 使用函数求奇数和50、习题5-3 使用函数计算两点间的距…...

git日常使用命令

实习这段时间使用了很多git指令来提交代码&#xff0c;简单记录一下日常使用的指令&#xff1a; 提交代码通常顺序&#xff1a; 1.git status 查看本地修改项 2.git add . 提交全部文件 &#xff08;这个 .是全部文件&#xff09;到暂存区 3.git commit -m ‘本次提交的说明’…...

ES6对象展开运算符浅拷贝or深拷贝

ES6中提出的对象展开运算符“…”就是用来展开元素的。有了它就不用代码循环遍历了&#xff0c;偷懒专用。 1. 合并数组 展开原有数组中的所有元素&#xff0c;可以合并成一个新的数组。 var a[1,2,3]; var b[4,5,6]; var c[...a,...b]; console.log(c) // 输出&#xff1a;…...

leaflet 上传包含shp的zip文件,在map上解析显示图形(059)

第059个 点击查看专栏目录 本示例的目的是介绍演示如何在vue+leaflet中本地上传包含shp的zip文件,利用shapefile读取shp数据,并在地图上显示图形。 直接复制下面的 vue+openlayers源代码,操作2分钟即可运行实现效果 文章目录 示例效果加载shapefile.js方式安装引用jszip(…...

CAN总线详细介绍

1.1 CAN是什么&#xff1f; CAN 最终成为国际标准 &#xff08; ISO11898(高速应用)和 ISO11519&#xff08;低速应用&#xff09;&#xff09;&#xff0c;是国际上应用最广泛的现场总线之一。 1.2 CAN总线特点 多主方式: 可以多主方式工作&#xff0c;网络上任意一个节点…...

python如何完成对 Excel文件的解密后读取?

通常为了防止重要的Excel文件数据内容的泄露&#xff0c;需要对文件整体进行加密与解密的操作。 对于文件的加解密过程&#xff0c;python也有很多非标准库来帮助我们完成操作&#xff0c;这里主要说明如何完成对Excel文件的解密与读取操作。 这里我们使用到的是msoffcrypto-…...

微服务实战--高级篇:RabbitMQ高级

服务异步通信-高级篇 消息队列在使用过程中&#xff0c;面临着很多实际问题需要思考&#xff1a; 1.消息可靠性 消息从发送&#xff0c;到消费者接收&#xff0c;会经理多个过程&#xff1a; 其中的每一步都可能导致消息丢失&#xff0c;常见的丢失原因包括&#xff1a; 发送…...

autoCAD2022 - 设置新的原点

文章目录autoCAD2022 - 设置新的原点概述笔记UCS原点设置功能的菜单位置ENDautoCAD2022 - 设置新的原点 概述 上次整板子的dxf时, 原来的原点不合适, 想调整一下. 当时整完了, 没记录. 这次用的时候, 又找半天… 设置新原点的功能, 不在顶部菜单中, 而是在视图右上角的UCS图标…...

spring boot 配置 mybatis-plus多数据源

简介Mybatis-puls 多数据源的使用&#xff0c;采用的是官方提供的dynamic-datasource-spring-boot-starter包的 DS 注解&#xff0c;具体可以参考官网&#xff1a;https://gitee.com/baomidou/dynamic-datasource-spring-boot-starterpom.xml文件引入如下依赖主要引入dynamic-d…...

独立产品灵感周刊 DecoHack #047 - 安卓手机上最有用的APP

本周刊记录有趣好玩的独立产品设计开发相关内容&#xff0c;每周发布&#xff0c;往期内容同样精彩&#xff0c;感兴趣的伙伴可以点击订阅我的周刊。为保证每期都能收到&#xff0c;建议邮件订阅。欢迎通过 Twitter 私信推荐或投稿。&#x1f4bb; 产品推荐 1. Bouncer Tempor…...

【面试题】JavaScript中递归的理解

大厂面试题分享 面试题库后端面试题库 &#xff08;面试必备&#xff09; 推荐&#xff1a;★★★★★地址&#xff1a;前端面试题库递归 RecursionTo iterate is human, to recurse, divine. 理解迭代&#xff0c;神理解递归。本文会以 JavaScript为主、有部分 Rust 举例说明。…...

PyTorch学习笔记

PyTorch学习笔记&#xff08;一&#xff09;&#xff1a;PyTorch环境安装 往期学习资料推荐&#xff1a; 1.Pytorch实战笔记_GoAI的博客-CSDN博客 2.Pytorch入门教程_GoAI的博客-CSDN博客 安装参考&#xff1a; 1.视频教程&#xff1a;3分钟深度学习【环境搭建】CUDA Anacon…...

HTML 语义化

目录 HTML 语义化HTML5 新特性HTML 语义化的好处语义化标签的使用场景最佳实践 HTML 语义化 HTML5 新特性 标准答案&#xff1a; 语义化标签&#xff1a; <header>&#xff1a;页头<nav>&#xff1a;导航<main>&#xff1a;主要内容<article>&#x…...

遍历 Map 类型集合的方法汇总

1 方法一 先用方法 keySet() 获取集合中的所有键。再通过 gey(key) 方法用对应键获取值 import java.util.HashMap; import java.util.Set;public class Test {public static void main(String[] args) {HashMap hashMap new HashMap();hashMap.put("语文",99);has…...

【快手拥抱开源】通过快手团队开源的 KwaiCoder-AutoThink-preview 解锁大语言模型的潜力

引言&#xff1a; 在人工智能快速发展的浪潮中&#xff0c;快手Kwaipilot团队推出的 KwaiCoder-AutoThink-preview 具有里程碑意义——这是首个公开的AutoThink大语言模型&#xff08;LLM&#xff09;。该模型代表着该领域的重大突破&#xff0c;通过独特方式融合思考与非思考…...

工业自动化时代的精准装配革新:迁移科技3D视觉系统如何重塑机器人定位装配

AI3D视觉的工业赋能者 迁移科技成立于2017年&#xff0c;作为行业领先的3D工业相机及视觉系统供应商&#xff0c;累计完成数亿元融资。其核心技术覆盖硬件设计、算法优化及软件集成&#xff0c;通过稳定、易用、高回报的AI3D视觉系统&#xff0c;为汽车、新能源、金属制造等行…...

Unit 1 深度强化学习简介

Deep RL Course ——Unit 1 Introduction 从理论和实践层面深入学习深度强化学习。学会使用知名的深度强化学习库&#xff0c;例如 Stable Baselines3、RL Baselines3 Zoo、Sample Factory 和 CleanRL。在独特的环境中训练智能体&#xff0c;比如 SnowballFight、Huggy the Do…...

Android15默认授权浮窗权限

我们经常有那种需求&#xff0c;客户需要定制的apk集成在ROM中&#xff0c;并且默认授予其【显示在其他应用的上层】权限&#xff0c;也就是我们常说的浮窗权限&#xff0c;那么我们就可以通过以下方法在wms、ams等系统服务的systemReady()方法中调用即可实现预置应用默认授权浮…...

第 86 场周赛:矩阵中的幻方、钥匙和房间、将数组拆分成斐波那契序列、猜猜这个单词

Q1、[中等] 矩阵中的幻方 1、题目描述 3 x 3 的幻方是一个填充有 从 1 到 9 的不同数字的 3 x 3 矩阵&#xff0c;其中每行&#xff0c;每列以及两条对角线上的各数之和都相等。 给定一个由整数组成的row x col 的 grid&#xff0c;其中有多少个 3 3 的 “幻方” 子矩阵&am…...

Maven 概述、安装、配置、仓库、私服详解

目录 1、Maven 概述 1.1 Maven 的定义 1.2 Maven 解决的问题 1.3 Maven 的核心特性与优势 2、Maven 安装 2.1 下载 Maven 2.2 安装配置 Maven 2.3 测试安装 2.4 修改 Maven 本地仓库的默认路径 3、Maven 配置 3.1 配置本地仓库 3.2 配置 JDK 3.3 IDEA 配置本地 Ma…...

苹果AI眼镜:从“工具”到“社交姿态”的范式革命——重新定义AI交互入口的未来机会

在2025年的AI硬件浪潮中,苹果AI眼镜(Apple Glasses)正在引发一场关于“人机交互形态”的深度思考。它并非简单地替代AirPods或Apple Watch,而是开辟了一个全新的、日常可接受的AI入口。其核心价值不在于功能的堆叠,而在于如何通过形态设计打破社交壁垒,成为用户“全天佩戴…...

【p2p、分布式,区块链笔记 MESH】Bluetooth蓝牙通信 BLE Mesh协议的拓扑结构 定向转发机制

目录 节点的功能承载层&#xff08;GATT/Adv&#xff09;局限性&#xff1a; 拓扑关系定向转发机制定向转发意义 CG 节点的功能 节点的功能由节点支持的特性和功能决定。所有节点都能够发送和接收网格消息。节点还可以选择支持一个或多个附加功能&#xff0c;如 Configuration …...