Linux信号sigaction / signal
Linux信号sigaction / signal
文章目录
- Linux信号sigaction / signal
- 目的
- 函数原型
- struct sigaction
- 信号枚举值
- ISO C99 signals.
- Historical signals specified by POSIX.
- New(er) POSIX signals (1003.1-2008, 1003.1-2013).
- Nonstandard signals found in all modern POSIX systems(including both BSD and Linux).
- 测试
- 1 处理信号后重启系统调用
- 2 处理信号后不重启系统调用
- 3 使用 sigaction 函数为 SIGUSR1 和 SIGUSR2 信号注册处理函数
- 实际代码框架
- 1. signal
- 2 将程序放入线程中
- 终端间通过kill发送信号
- Reference
目的
我们可以通过sigaction函数或者signal指定某种特定信号,在程序执行过程中,通过发送信号,达到改变程序运行状态的目的,比如让程序停止等。
函数原型
获取或者设定与指定信号相关联的处理动作。
/* Get and/or set the action for signal SIG. */
extern int sigaction (int __sig, const struct sigaction *__restrict __act,struct sigaction *__restrict __oact) __THROW;extern __sighandler_t signal (int __sig, __sighandler_t __handler)__THROW;
sigaction函数执行信号处理程序时,会把当前信号加入到进程的信号屏蔽字中,从而防止在进行信号处理期间信号丢失。

对signal(),Linux默认会自动重启动被中断的系统调用;
而对于 sigaction(),Linux默认并不会自动重启动,所以如果希望执行信号处理后自动重启动先前中断的系统调用,就需要为sa_flags指定 SA_RESTART标志。
struct sigaction
当信号到达时,用于描述采取的动作的结构
/* Structure describing the action to be taken when a signal arrives. */
struct sigaction{/* Signal handler. */
#if defined __USE_POSIX199309 || defined __USE_XOPEN_EXTENDEDunion{/* Used if SA_SIGINFO is not set. */__sighandler_t sa_handler;/* Used if SA_SIGINFO is set. */void (*sa_sigaction) (int, siginfo_t *, void *);}__sigaction_handler;
# define sa_handler __sigaction_handler.sa_handler
# define sa_sigaction __sigaction_handler.sa_sigaction
#else__sighandler_t sa_handler;
#endif/* Additional set of signals to be blocked. */__sigset_t sa_mask;/* Special flags. */int sa_flags;/* Restore handler. */void (*sa_restorer) (void);};
- sa_handler 和signal()的参数__handler相同,代表新的信号处理函数
- sa_mask 表示处理信号时,需要阻止的信号集
- sa_flags 处理信号时的标志位
- SA_RESETHAND:在调用处理函数前,将信号的处理函数重置为缺省值(即SIG_IGN)
- SA_RESTART:执行信号处理后,自动重启先前中断的系统调用
- SA_NODEFER :一般情况下, 当信号处理函数运行时,内核将阻塞该给定信号。但是如果设置了 SA_NODEFER标记, 那么在该信号处理函数运行时,内核将不会阻塞该信号
- SA_ONSTACK:使用由sigaltstack()安装的备选栈
- SA_SIGINFO:调用信号处理程序时携带了额外参数(提供了关于信号的深入信息)
信号枚举值
ISO C99 signals.
| 符号 | 数值 | 含义 |
|---|---|---|
| SIGINT | 2 | Interactive attention signal. |
| SIGILL | 4 | Illegal instruction. |
| SIGABRT | 6 | Abnormal termination. |
| SIGFPE | 8 | Erroneous arithmetic operation. |
| SIGSEGV | 11 | Invalid access to storage. |
| SIGTERM | 15 | Termination request. |
Historical signals specified by POSIX.
| 符号 | 数值 | 含义 |
|---|---|---|
| SIGHUP | 1 | Hangup. |
| SIGQUIT | 3 | Quit. |
| SIGTRAP | 5 | Trace/breakpoint trap. |
| SIGKILL | 9 | Killed. |
| SIGBUS | 10 | Bus error. |
| SIGSYS | 12 | Bad system call. |
| SIGPIPE | 13 | Broken pipe. |
| SIGALRM | 14 | Alarm clock. |
New(er) POSIX signals (1003.1-2008, 1003.1-2013).
| 符号 | 数值 | 含义 |
|---|---|---|
| SIGURG | 16 | Urgent data is available at a socket. |
| SIGSTOP | 17 | Stop, unblockable. |
| SIGTSTP | 18 | Keyboard stop. |
| SIGCONT | 19 | Continue. |
| SIGCHLD | 20 | Child terminated or stopped. |
| SIGTTIN | 21 | Background read from control terminal. |
| SIGTTOU | 22 | Background write to control terminal. |
| SIGPOLL | 23 | Pollable event occurred (System V). |
| SIGXCPU | 24 | CPU time limit exceeded. |
| SIGXFSZ | 25 | File size limit exceeded. |
| SIGVTALRM | 26 | Virtual timer expired. |
| SIGPROF | 27 | Profiling timer expired. |
| SIGUSR1 | 30 | User-defined signal 1. |
| SIGUSR2 | 31 | User-defined signal 2. |
Nonstandard signals found in all modern POSIX systems(including both BSD and Linux).
| 符号 | 数值 | 含义 |
|---|---|---|
| SIGWINCH | 28 | Window size change (4.3 BSD, Sun). |
测试
1 处理信号后重启系统调用
#include <iostream>
#include "stdio.h"
#include <signal.h>
#include <stdlib.h>void signalHandle(int signum) {if(signum = SIGINT) {std::cout << "SIGINT recived" << std::endl;}
}int main(void){struct sigaction act;sigemptyset(&act.sa_mask);//这里使用SA_RESTART执行信号处理后自动重启到先前中断的系统调用,可以多次捕捉信号act.sa_flags = (SA_SIGINFO|SA_RESTART);act.sa_handler = signalHandle;sigaction(SIGINT, &act, NULL);while(1){}
}
当按下 CTRL + c时,会一直打印。
$ ./main
^CSIGINT recived
^CSIGINT recived
^CSIGINT recived
2 处理信号后不重启系统调用
#include <iostream>
#include <stdio.h>
#include <unistd.h>
#include <signal.h>
#include <errno.h>void show_handler(int sig)
{printf("I got signal %dn\n", sig);int i;for(i = 0; i < 5; i++) {printf("i = %dn\n", i);sleep(1);}
}int main(void)
{// section 1int i = 0;struct sigaction act, oldact;act.sa_handler = show_handler;sigaddset(&act.sa_mask, SIGQUIT); act.sa_flags = SA_RESETHAND | SA_NODEFER; //act.sa_flags = 0; sigaction(SIGINT, &act, &oldact);while(1) {sleep(1);printf("sleeping %dn\n", i);i++;}return 0;
}
程序起来后,控制台输入CTRL + c,能够接收到信号。再次输入时,程序退出。
(base) qiancj@qiancj-Dell-G15-5510:~/codes/test/build$ ./main
sleeping 0
sleeping 1
^CI got signal 2
i = 0
i = 1
i = 2
i = 3
i = 4
sleeping 2
^C
(base) qiancj@qiancj-Dell-G15-5510:~/codes/test/build$
3 使用 sigaction 函数为 SIGUSR1 和 SIGUSR2 信号注册处理函数
#include <iostream>
#include <stdio.h>
#include <unistd.h>
#include <signal.h>
#include <errno.h>static void sig_usr(int signum)
{if(signum == SIGUSR1){printf("SIGUSR1 received\n");}else if(signum == SIGUSR2){printf("SIGUSR2 received\n");}else{printf("signal %d received\n", signum);}
}int main(void)
{
// section 2char buf[512];int n;struct sigaction sa_usr;sa_usr.sa_flags = 0;sa_usr.sa_handler = sig_usr; //信号处理函数sigaction(SIGUSR1, &sa_usr, NULL);sigaction(SIGUSR2, &sa_usr, NULL);printf("My PID is %dn\n", getpid());while(1){if((n = read(STDIN_FILENO, buf, 511)) == -1){if(errno == EINTR){printf("read is interrupted by signal\n");}}else{buf[n] = '0';printf("%d bytes read: %s\n", n, buf);}}return 0;
}
当一个终端启动程序,获取当前PID号(13770),另一个终端输入kill -USR1 32230或者kill -USR2 32230,第一个终端就会收到中断信号
(base) qiancj@qiancj-Dell-G15-5510:~/codes/test/build$ ./main
My PID is 32230nSIGUSR1 received
read is interrupted by signal
SIGUSR2 received
read is interrupted by signal
实际代码框架
当程序需要一直运行时,需要人为中断停止时,可以使用信号函数,具体框架可以设置为以下2种:
1. signal
#include <csignal>
#include <iostream>
#include <sstream>
#include <string>
#include <thread>sig_atomic_t stop_flag = 0;void INTSigHandler(int32_t num) {stop_flag = 1;std::cout << " Signal Interactive attention " << num << " received."<< std::endl;
}int main(int argc, char **argv) {signal(SIGINT, INTSigHandler);// work always.......while ((!stop_flag)) {LOG_INFO << "I am working...";std::this_thread::sleep_for(std::chrono::seconds(1));}// Stop working! Go to rest!return 0;
}
2 将程序放入线程中
#include <chrono>
#include <csignal>
#include <cstdint>
#include <cstdio>
#include <cstdlib>namespace {std::atomic_bool continueExecution{true};void SigTermHandler(int signal) {if (signal == SIGTERM || signal == SIGINT) {// set atomic exit flagcontinueExecution = false;}
}bool RegisterSigTermHandler() {struct sigaction sa;sa.sa_handler = SigTermHandler;sa.sa_flags = 0;sigemptyset(&sa.sa_mask);// register signal handlerif (sigaction(SIGTERM, &sa, NULL) == -1) {// Could not register a SIGTERM signal handlerreturn false;}if (sigaction(SIGINT, &sa, NULL) == -1) {// Could not register a SIGTERM signal handlerreturn false;}return true;
}} // namespacevoid ThreadAct1();void ThreadAct1() {while (continueExecution) {// always work......// sleepstd::this_thread::sleep_for(std::chrono::milliseconds(10000));}
}int main(int argc, char *argv[]) {if (!RegisterSigTermHandler()) {std::cout << "Unable to register signal handler" << std::endl;}// start threadstd::thread act1(ThreadAct1);act1.join();return 0;
}
终端间通过kill发送信号
(base) qiancj@qiancj-Dell-G15-5510:~$ kill --help
kill: kill [-s sigspec | -n signum | -sigspec] pid | jobspec ... or kill -l [sigspec]Send a signal to a job.Send the processes identified by PID or JOBSPEC the signal named bySIGSPEC or SIGNUM. If neither SIGSPEC nor SIGNUM is present, thenSIGTERM is assumed.Options:-s sig SIG is a signal name-n sig SIG is a signal number-l list the signal names; if arguments follow `-l' they areassumed to be signal numbers for which names should be listed-L synonym for -lKill is a shell builtin for two reasons: it allows job IDs to be usedinstead of process IDs, and allows processes to be killed if the limiton processes that you can create is reached.Exit Status:Returns success unless an invalid option is given or an error occurs.
信号种类
(base) qiancj@qiancj-Dell-G15-5510:~$ kill -l1) SIGHUP 2) SIGINT 3) SIGQUIT 4) SIGILL 5) SIGTRAP6) SIGABRT 7) SIGBUS 8) SIGFPE 9) SIGKILL 10) SIGUSR1
11) SIGSEGV 12) SIGUSR2 13) SIGPIPE 14) SIGALRM 15) SIGTERM
16) SIGSTKFLT 17) SIGCHLD 18) SIGCONT 19) SIGSTOP 20) SIGTSTP
21) SIGTTIN 22) SIGTTOU 23) SIGURG 24) SIGXCPU 25) SIGXFSZ
26) SIGVTALRM 27) SIGPROF 28) SIGWINCH 29) SIGIO 30) SIGPWR
31) SIGSYS 34) SIGRTMIN 35) SIGRTMIN+1 36) SIGRTMIN+2 37) SIGRTMIN+3
38) SIGRTMIN+4 39) SIGRTMIN+5 40) SIGRTMIN+6 41) SIGRTMIN+7 42) SIGRTMIN+8
43) SIGRTMIN+9 44) SIGRTMIN+10 45) SIGRTMIN+11 46) SIGRTMIN+12 47) SIGRTMIN+13
48) SIGRTMIN+14 49) SIGRTMIN+15 50) SIGRTMAX-14 51) SIGRTMAX-13 52) SIGRTMAX-12
53) SIGRTMAX-11 54) SIGRTMAX-10 55) SIGRTMAX-9 56) SIGRTMAX-8 57) SIGRTMAX-7
58) SIGRTMAX-6 59) SIGRTMAX-5 60) SIGRTMAX-4 61) SIGRTMAX-3 62) SIGRTMAX-2
63) SIGRTMAX-1 64) SIGRTMAX
Reference
- linux中sigaction函数详解
- sigaction详细解析
相关文章:
Linux信号sigaction / signal
Linux信号sigaction / signal 文章目录Linux信号sigaction / signal目的函数原型struct sigaction信号枚举值ISO C99 signals.Historical signals specified by POSIX.New(er) POSIX signals (1003.1-2008, 1003.1-2013).Nonstandard signals found in all modern POSIX system…...
坦克大战第一阶段代码
package tanke.game;import javax.swing.*; import java.awt.*; import java.awt.event.KeyEvent; import java.awt.event.KeyListener; import java.util.Vector;//为了监听键盘事件,实现keylistener public class mypanel extends JPanel implements KeyListener …...
博客系统前端实现
目录 1.预期效果 2.实现博客列表页 3.实现博客正文页 4.实现博客登录页 5.实现博客编辑页面 1.预期效果 对前端html,css,js有大致的了解后,现在我们实现了一个博客系统的前端页面.一共分为四个页面没分别是:登陆页面,博客列表页,博客正文页,博客编辑页 我们看下四个界面…...
ChatGPT技术原理、研究框架,应用实践及发展趋势(附166份报告)
一、AI框架重要性日益突显,框架技术发展进入繁荣期,国内AI框架技术加速发展: 1、AI框架作为衔接数据和模型的重要桥梁,发展进入繁荣期,国内外框架功能及性能加速迭代; 2、Pytorch、Tensorflow占据AI框…...
【屏幕自适应页面适配问题】CSS的@media,为了适应1440×900的屏幕,使用@media解决问题
文章目录bug修改实例CSS3 media 查询CSS 多媒体查询,适配各种设备尺寸bug修改实例 <template><div id"deptAllDown" style"height: 400px;width:880px"/> </template>为了适应1440900的屏幕,使用media解决问题 …...
一篇文章理解堆栈溢出
一篇文章理解堆栈溢出引言栈溢出ret2text答案ret2shellcode答案ret2syscall答案栈迁移答案堆溢出 unlink - UAF堆结构小提示向前合并/向后合并堆溢出题答案引言 让新手快速理解堆栈溢出,尽可能写的简单一些。 栈溢出 代码执行到进入函数之前都会记录返回地址到SP…...
优化模型验证关键代码27:多旅行商问题的变体-多起点单目的地问题和多汉密尔顿路径问题
目录 1 多起点单目的地问题(Multiple departures single destination mTSP) 1.1 符号列表 1.2 数学模型 1.4 解的可视化结果...
快速搭建第一个SpringCloud程序
目录 1、Spring Boot项目脚手架快速搭建 1.1 生成工程基本配置 1.2 生成工程。 1.3 导入开发工具(此处为Idea) 1.4 运行代码 1.5 验证是否能访问 2、Spring Cloud环境搭建 2.1 版本匹配问题 2.2 Spring Cloud环境测试 3、引入Eureka Server 3…...
【离散数学】图论
1、有n个点没有边 零图 2、有1个点没有边 平凡图 3、含有平行边的图 多重图 4、简单图 不含有平行边和自回环的图 5、任意两个结点之间都有边 完全图 6、环贡献 两度 7、所有顶点的度数之和等于边数的两倍 8、在有向图中所有顶点的出度之和 或者 入度之和 等于边数 9、度数为…...
代码随想录算法训练营第三十七天-贪心算法6| 738.单调递增的数字 968.监控二叉树 总结
738.单调递增的数字 贪心算法 题目要求小于等于N的最大单调递增的整数,那么拿一个两位的数字来举例。 例如:98,一旦出现strNum[i - 1] > strNum[i]的情况(非单调递增),首先想让strNum[i - 1]--&#…...
【Linux】线程中的互斥锁、条件变量、信号量(数据安全问题、生产消费模型、阻塞队列和环形队列的实现)
文章目录1、线程互斥1.1 线程间频繁切换导致的问题1.2 使用互斥锁1.3 互斥锁的原理1.4 线程中的数据安全问题2、线程同步之条件变量2.1 生产消费模型2.2 条件变量概念和调用函数2.3 阻塞队列的实现3、线程同步之信号量3.1 理解信号量3.2 信号量接口3.3 环形队列的实现4、小结1、…...
MySQL8.0的安装和配置
🎉🎉🎉点进来你就是我的人了 博主主页:🙈🙈🙈戳一戳,欢迎大佬指点!人生格言:当你的才华撑不起你的野心的时候,你就应该静下心来学习! 欢迎志同道合的朋友一起加油喔🦾&am…...
LinuxGUI自动化测试框架搭建(三)-虚拟机安装(Hyper-V或者VMWare)
(三)-虚拟机安装(Hyper-V或者VMWare)1 Hyper-V安装1.1 方法一:直接启用1.2 方法二:下载安装1.3 打开Hyper-V2 VMWare安装注意:Hyper-V或者VMWare只安装一个,只安装一个,只…...
改进YOLO系列:数据增强扩充(有增强图像和标注),包含copypaste、翻转、cutout等八种增强方式
这里写目录标题 一、简介二、数据增强方法介绍复制-粘贴(Copy-paste)翻转(Flip)Cutout加噪声(Noise)亮度调整(Brightness)平移(Shift)旋转(Rotation)裁剪(Crop)copy-paste的代码一、简介 数据增强是一种通过对原始数据进行随机变换、扰动等操作来生成新的训练样…...
c++11 标准模板(STL)(std::stack)(一)
定义于头文件 <stack> template< class T, class Container std::deque<T> > class stack;std::stack 类是容器适配器,它给予程序员栈的功能——特别是 FILO (先进后出)数据结构。 该类模板表现为底层容器的包装…...
C++-c语言词法分析器
一、运行截图 对于 Test.c 的词法分析结果 对于词法分析器本身的源代码的分析结果 二、主要功能 经过不断的修正和测试代码,分析测试结果,该词法分析器主要实现了以下功能: 1. 识别关键字 实验要求:if else while do for main…...
Maven工具复习
Maven从入门到放弃Maven概述Maven 的配置Maven的基本使用IDEA 配置MAVENMaven坐标IDEA 创建MavenIDEA 导入Maven关于右侧Maven小标签(也就是Maven面板)找不到问题的解决办法关于不小心把IDEA主菜单搞消失的解决办法依赖管理Maven概述 Maven是一个工具提供了一套标准的项目结构…...
算法总结-深度优先遍历和广度优先遍历
深度优先遍历(Depth First Search,简称DFS) 与广度优先遍历(Breath First Search,简称BFS)是图论中两种非常重要的算法,生产上广泛用于拓扑排序,寻路(走迷宫),搜索引擎,爬虫等。 一、深度优先遍历 深度优先…...
【Linux】Centos安装mvn命令(maven)
🍁博主简介 🏅云计算领域优质创作者 🏅华为云开发者社区专家博主 🏅阿里云开发者社区专家博主 💊交流社区:运维交流社区 欢迎大家的加入! 文章目录一、下载maven包方法一:官…...
驱动保护 -- 通过PID保护指定进程
一、设计界面 1、添加一个编辑框输入要保护的进程PID,并添加两个按钮,一个保护进程,一个解除保护 2、右击编辑框,添加变量 二、驱动层代码实现 1、声明一个受保护的进程PID数组 static UINT32 受保护的进程PID[256] { 0 }; 2…...
【Axure高保真原型】引导弹窗
今天和大家中分享引导弹窗的原型模板,载入页面后,会显示引导弹窗,适用于引导用户使用页面,点击完成后,会显示下一个引导弹窗,直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…...
网络六边形受到攻击
大家读完觉得有帮助记得关注和点赞!!! 抽象 现代智能交通系统 (ITS) 的一个关键要求是能够以安全、可靠和匿名的方式从互联车辆和移动设备收集地理参考数据。Nexagon 协议建立在 IETF 定位器/ID 分离协议 (…...
【根据当天日期输出明天的日期(需对闰年做判定)。】2022-5-15
缘由根据当天日期输出明天的日期(需对闰年做判定)。日期类型结构体如下: struct data{ int year; int month; int day;};-编程语言-CSDN问答 struct mdata{ int year; int month; int day; }mdata; int 天数(int year, int month) {switch (month){case 1: case 3:…...
如何在看板中体现优先级变化
在看板中有效体现优先级变化的关键措施包括:采用颜色或标签标识优先级、设置任务排序规则、使用独立的优先级列或泳道、结合自动化规则同步优先级变化、建立定期的优先级审查流程。其中,设置任务排序规则尤其重要,因为它让看板视觉上直观地体…...
java调用dll出现unsatisfiedLinkError以及JNA和JNI的区别
UnsatisfiedLinkError 在对接硬件设备中,我们会遇到使用 java 调用 dll文件 的情况,此时大概率出现UnsatisfiedLinkError链接错误,原因可能有如下几种 类名错误包名错误方法名参数错误使用 JNI 协议调用,结果 dll 未实现 JNI 协…...
全球首个30米分辨率湿地数据集(2000—2022)
数据简介 今天我们分享的数据是全球30米分辨率湿地数据集,包含8种湿地亚类,该数据以0.5X0.5的瓦片存储,我们整理了所有属于中国的瓦片名称与其对应省份,方便大家研究使用。 该数据集作为全球首个30米分辨率、覆盖2000–2022年时间…...
20个超级好用的 CSS 动画库
分享 20 个最佳 CSS 动画库。 它们中的大多数将生成纯 CSS 代码,而不需要任何外部库。 1.Animate.css 一个开箱即用型的跨浏览器动画库,可供你在项目中使用。 2.Magic Animations CSS3 一组简单的动画,可以包含在你的网页或应用项目中。 3.An…...
【C++进阶篇】智能指针
C内存管理终极指南:智能指针从入门到源码剖析 一. 智能指针1.1 auto_ptr1.2 unique_ptr1.3 shared_ptr1.4 make_shared 二. 原理三. shared_ptr循环引用问题三. 线程安全问题四. 内存泄漏4.1 什么是内存泄漏4.2 危害4.3 避免内存泄漏 五. 最后 一. 智能指针 智能指…...
day36-多路IO复用
一、基本概念 (服务器多客户端模型) 定义:单线程或单进程同时监测若干个文件描述符是否可以执行IO操作的能力 作用:应用程序通常需要处理来自多条事件流中的事件,比如我现在用的电脑,需要同时处理键盘鼠标…...
Modbus RTU与Modbus TCP详解指南
目录 1. Modbus协议基础 1.1 什么是Modbus? 1.2 Modbus协议历史 1.3 Modbus协议族 1.4 Modbus通信模型 🎭 主从架构 🔄 请求响应模式 2. Modbus RTU详解 2.1 RTU是什么? 2.2 RTU物理层 🔌 连接方式 ⚡ 通信参数 2.3 RTU数据帧格式 📦 帧结构详解 🔍…...
