什么是信号处理?如何处理信号?
C语言信号处理详解
第一部分:什么是信号?
信号是一种进程间通信的机制,用于通知进程发生了某种事件或异常情况。在C语言中,信号是一种软件中断,它可以被操作系统或其他进程发送给目标进程。每个信号都有一个唯一的数字标识符,称为信号编号(Signal Number)。例如,常见的信号包括SIGINT(中断进程)、SIGTERM(终止进程)、SIGSEGV(段错误)等。
信号可以用于以下几种情况:
-
进程间通信:一个进程可以向另一个进程发送信号,以通知它某个事件的发生或请求其执行某个操作。
-
异常处理:操作系统可以向进程发送信号,以通知它发生了某个异常情况,例如除零错误、段错误等。
-
用户交互:用户可以通过键盘或终端发送信号来与正在运行的程序交互,例如使用Ctrl+C发送SIGINT信号来中断程序的执行。
第二部分:信号的基本操作
2.1 发送信号
在C语言中,可以使用kill
函数或raise
函数来向目标进程发送信号。
#include <signal.h>int kill(pid_t pid, int sig);
int raise(int sig);
-
kill
函数用于向指定的进程发送信号sig
,pid
参数指定了目标进程的进程ID。如果pid
为正数,则表示发送信号给进程ID为pid
的进程;如果pid
为0,则表示发送信号给当前进程所在进程组的所有进程;如果pid
为-1,则表示发送信号给当前用户的所有进程;如果pid
小于-1,则表示发送信号给进程组ID等于pid
的所有进程。 -
raise
函数用于向当前进程发送信号sig
。
2.2 接收信号
要在C程序中接收信号,可以使用signal
函数或sigaction
函数来注册信号处理函数。
2.2.1 signal
函数
signal
函数用于注册信号处理函数,其原型如下:
#include <signal.h>void (*signal(int sig, void (*handler)(int)))(int);
sig
参数指定了要处理的信号,例如SIGINT
、SIGTERM
等。handler
参数是一个函数指针,指向处理该信号的函数。
以下是一个使用signal
函数注册信号处理函数的示例:
#include <stdio.h>
#include <signal.h>
#include <unistd.h>void sigint_handler(int signo) {printf("Received SIGINT signal (%d).\n", signo);
}int main() {// 注册SIGINT信号处理函数signal(SIGINT, sigint_handler);while (1) {sleep(1); // 模拟程序执行}return 0;
}
在上面的示例中,当程序接收到Ctrl+C信号(SIGINT)时,将调用sigint_handler
函数来处理该信号。
2.2.2 sigaction
函数
sigaction
函数提供了更加灵活的信号处理方式,其原型如下:
#include <signal.h>int sigaction(int sig, const struct sigaction *restrict act, struct sigaction *restrict oldact);
sig
参数指定了要处理的信号,例如SIGINT
、SIGTERM
等。act
参数是一个指向struct sigaction
结构的指针,用于设置信号处理的行为。oldact
参数是一个指向struct sigaction
结构的指针,用于获取之前的信号处理行为。
struct sigaction
结构定义如下:
struct sigaction {void (*sa_handler)(int); // 信号处理函数sigset_t sa_mask; // 信号屏蔽字集合int sa_flags; // 信号处理标志void (*sa_sigaction)(int, siginfo_t *, void *); // 信号处理函数(扩展)
};
以下是一个使用sigaction
函数注册信号处理函数的示例:
#include <stdio.h>
#include <signal.h>
#include <unistd.h>void sigint_handler(int signo) {printf("Received SIGINT signal (%d).\n", signo);
}int main() {struct sigaction sa;sa.sa_handler = sigint_handler;sa.sa_flags = 0;sigemptyset(&sa.sa_mask);// 注册SIGINT信号处理函数sigaction(SIGINT, &sa, NULL);while (1) {sleep(1); // 模拟程序执行}return 0;
}
sigaction
函数允许更灵活地控制信号处理,可以设置额外的标志和信号屏蔽。
2.3 信号默认操作
每个信号都有一个默认操作,例如终止进程、中断进程等。可以使用signal
函数的第二个参数来指定信号处理函数,或者将信号处理函数设置为SIG_IGN
(忽略信号)或SIG_DFL
(恢复默认操作)。
以下是一些常见的信号默认操作:
SIGINT
(Ctrl+C)默认操作是中断进程。SIGTERM
默认操作是终止进程。SIGQUIT
(Ctrl+\)默认操作是终止进程并生成核心转储文件。SIGHUP
默认操作是终止进程,通常用于重新加载配置。SIGKILL
默认操作是强制终止进程,无法被捕获或忽略。
第三部分:信号处理函数
信号处理函数是用户自定义的函数,用于处理特定信号的发生。信号处理函数的原型通常是:
void handler_function(int signo);
signo
参数指定了触发信号处理函数的信号编号。
信号处理函数可以执行各种操作,例如记录日志、清理资源、继续执行等。然而,由于信号处理函数在信号发生时异步执行,因此需要谨慎编写,避免使用不可重入函数、全局变量等可能导致不确定行为的操作。
以下是一个示例,演示如何编写一个简单的信号处理函数:
#include <stdio.h>
#include <signal.h>
#include <unistd.h>void sigint_handler(int signo) {printf("Received SIGINT signal (%d).\n", signo);
}int main() {// 注册SIGINT信号处理函数signal(SIGINT, sigint_handler);while (1) {sleep(1); // 模拟程序执行}return 0;
}
在上面的示例中,sigint_handler
函数用于处理SIGINT信号的发生,它简单地打印一条消息。
第四部分:信号的处理方式
信号的处理方式分为以下几种:
4.1 忽略信号
可以通过将信号处理函数设置为SIG_IGN
来忽略信号。例如,要忽略SIGINT
信号,可以这样做:
#include <signal.h>int main() {signal(SIGINT, SIG_IGN); // 忽略SIGINT信号while (1) {// 程序不会响应Ctrl+C}return 0;
}
4.2 捕获信号
通过注册信号处理函数,可以捕获信号并在发生时执行特定操作。示例如前面所示。
4.3 恢复默认操作
可以将信号处理函数设置为SIG_DFL
,以恢复信号的默认操作。例如,要恢复SIGINT
信号的默认操作,可以这样做:
#include <signal.h>int main() {signal(SIGINT, SIG_DFL); // 恢复SIGINT信号的默认操作while (1) {// Ctrl+C将中断进程}return 0;
}
4.4 阻塞信号
可以使用sigprocmask
函数来阻塞或解除阻塞信号。阻塞信号意味着信号将被排队,不会立即传递给进程。当信号解除阻塞时,排队的信号将被传递给进程。
#include <stdio.h>
#include <signal.h>
#include <unistd.h>void sigint_handler(int signo) {printf("Received SIGINT signal (%d).\n", signo);
}int main() {sigset_t mask;sigemptyset(&mask);sigaddset(&mask, SIGINT);// 阻塞SIGINT信号sigprocmask(SIG_BLOCK, &mask, NULL);// 注册SIGINT信号处理函数signal(SIGINT, sigint_handler);while (1) {sleep(1);printf("Working...\n");}return 0;
}
在上面的示例中,sigprocmask
函数用于阻塞SIGINT信号,直到解除阻塞时才会执行信号处理函数。
4.5 发送信号
前面已经介绍了如何使用kill
函数或raise
函数来向进程发送信号。
第五部分:信号处理的注意事项
5.1 信号不可靠性
信号处理是一种异步操作,因此存在信号不可靠性的问题。例如,如果在两次信号之间处理函数没有完成,第二个信号可能会丢失。因此,在信号处理函数中应谨慎使用全局变量、不可重入函数等。
5.2 信号重入
信号处理函数应该是可重入的,即可以在信号处理函数执行期间再次接收到相同信号而不会导致问题。要实现可重入性,可以使用sigaction
函数注册信号处理函数,同时在信号处理函数中避免使用全局变量和不可重入函数。
5.3 信号与系统调用
在系统调用期间,通常会将信号屏蔽(阻塞),以避免在关键操作期间接收到信号导致不一致性。一些系统调用会自动恢复信号屏蔽,但一些不会。因此,在系统调用中要注意信号处理的状态。
5.4 信号与多线程
多线程程序中,每个线程都有自己的信号屏蔽状态。默认情况下,新线程会继承创建它的线程的信号屏蔽状态。因此,在多线程程序中要小心管理信号屏蔽状态,以确保线程间的信号不会相互干扰。
第六部分:总结
信号处理是C语言中处理异步事件和异常情况的重要机制。本文介绍了信号的基本概念、信号的发送和接收、信号处理函数的编写方式、信号的处理方式以及注意事项。了解信号处理可以帮助程序员更好地处理各种情况下的信号,提高程序的健壮性和可靠性。在实际编程中,要谨慎处理信号,避免不可预测的行为,保证程序的稳定性。
相关文章:
什么是信号处理?如何处理信号?
C语言信号处理详解 第一部分:什么是信号? 信号是一种进程间通信的机制,用于通知进程发生了某种事件或异常情况。在C语言中,信号是一种软件中断,它可以被操作系统或其他进程发送给目标进程。每个信号都有一个唯一的数…...

谈谈 Redis 数据类型底层的数据结构?
谈谈 Redis 数据类型底层的数据结构? RedisObject 在 Redis 中,redisObject 是一个非常重要的数据结构,它用于保存字符串、列表、集合、哈希表和有序集合等类型的值。以下是关于 redisObject 结构体的定义: typedef struct redisObject {…...

九、GC收集日志
JVM由浅入深系列一、关于Java性能的误解二、Java性能概述三、了解JVM概述四、探索JVM架构五、垃圾收集基础六、HotSpot中的垃圾收集七、垃圾收集中级八、垃圾收集高级👋GC收集日志 ⚽️1. 认识GC收集日志 垃圾收集日志是一个重要的信息来源,对于与性能相关的一些悬而未决的…...

SimpleCG动画示例--汉诺塔动画演示
前言 SimpleCG的使用方法在前面已经介绍了许多,有兴趣的同学如果有去动手,制作一些简单动画应该没多大问题的。所以这次我们来演示一下简单动画。我们刚学习C语言的递归函数时,有一个经典例子相信很多同学都写过,那就是汉诺塔。那…...
反弹shell脚本(php-reverse-shell)
平时经常打靶机 这里贴一个 反弹shell的脚本 <?php // php-reverse-shell - A Reverse Shell implementation in PHP // Copyright (C) 2007 pentestmonkeypentestmonkey.net // // This tool may be used for legal purposes only. Users take full responsibility // f…...

XSS-labs
XSS常见的触发标签_xss标签_H3rmesk1t的博客-CSDN博客 该补习补习xss漏洞了 漏洞原理 网站存在 静态 和 动态 网站 xss 针对的网站 就是 动态网站 动态网站会根据 用户的环境 与 需求 反馈出 不同的响应静态页面 代码写死了 只会存在代码中有的内容 通过动态网站 用户体…...

C++简单实现AVL树
目录 一、AVL树的概念 二、AVL树的性质 三、AVL树节点的定义 四、AVL树的插入 4.1 parent的平衡因子为0 4.2 parent的平衡因子为1或-1 4.3 parent的平衡因子为2或-2 4.3.1 左单旋 4.3.2 右单旋 4.3.3 先左单旋再右单旋 4.3.4 先右单旋再左单旋 4.4 插入节点完整代码…...

UE4 Cesium 与ultra dynamic sky插件天气融合
晴天: 雨天: 雨天湿度: 小雪: 中雪: 找到该路径这个材质: 双击点开: 将Wet_Weather_Effects与Snow_Weather_Effects复制下来,包括参数节点 找到该路径这个材质,双击点开&…...

SpringCloud Gateway--Predicate/断言(详细介绍)下
😀前言 本篇博文是关于SpringCloud Gateway–Predicate/断言(详细介绍)下,希望你能够喜欢 🏠个人主页:晨犀主页 🧑个人简介:大家好,我是晨犀,希望我的文章可以…...

SOC芯片学习--GPIO简介
原创 硬件设计技术 硬件设计技术 2023-07-20 00:04 发表于广东 收录于合集#集成电路--IC7个 一、GPIO定义、分类: GPIO(英语:General-purpose input/output),通用型之输入输出的简称,其接脚可以供使用者由…...

skywalking源码本地编译运行经验总结
前言 最近工作原因在弄skywalking,为了进一步熟悉拉了代码下来准备debug,但是编译启动项目我就费了老大劲了,所以准备写这篇,帮兄弟们少踩点坑。 正确步骤 既然是用开源的东西,那么最好就是按照人家的方式使用&…...

K8s架构简述
以部署一个nginx服务说明kubernetes系统各个组件调用关系: 一旦kubernetes环境启动之后,master和node都会将自身的信息存储到etcd数据库中 一个nginx服务的安装请求会首先被发送到master节点的apiServer组件 apiServer组件会调用scheduler组件来决定到底…...
linkedlist和arraylist的区别
LinkedList和ArrayList都是常见的数据结构,用于存储和操作集合元素,如果需要频繁进行插入和删除操作,LinkedList可能更适合。如果需要快速随机访问和较小的内存占用,ArrayList可能更合适。 以下是它们之间存在一些关键的区别&…...

[尚硅谷React笔记]——第2章 React面向组件编程
目录: 基本理解和使用: 使用React开发者工具调试函数式组件复习类的基本知识类式组件组件三大核心属性1: state 复习类中方法this指向: 复习bind函数:解决changeWeather中this指向问题:一般写法:state.htm…...

嵌入式学习笔记(40)看门狗定时器
7.5.1什么是看门狗、有何用 (1)看门狗定时器和普通定时器并无本质区别。定时器可以设定一个时间,在这个时间完成之前定时器不断计时,时间到的时候定时器会复位CPU(重启系统)。 (2)系统正常工作的时候当然不希望被重启࿰…...

点击、拖拉拽,BI系统让业务掌握数据分析主动权
在今天的商业环境中,数据分析已经成为企业获取竞争优势的关键因素之一。然而,许多企业在面对复杂的数据分析工具时,却常常感到困扰。这些工具往往需要专业的技术人员操作,而且界面复杂,难以理解和使用。对业务人员来说…...
C++模拟题[第一周-T1] 扑克
[第一周-T1] 扑克 题目描述 斗地主是一种使用 A \tt A A 到 K \tt K K 加上大小王的共 54 54 54 张扑克牌来进行的游戏,其中大小王各一张,其它数码牌各四张。在斗地主中,牌的大小关系根据牌的数码表示如下: 3 < 4 < 5 …...

ciscn_2019_s_9
ciscn_2019_s_9 Arch: i386-32-little RELRO: Partial RELRO Stack: No canary found NX: NX disabled PIE: No PIE (0x8048000) RWX: Has RWX segments32位,啥也没开,开心愉悦写shellcode int pwn() {char s[24]; // [esp8…...

微信、支付宝、百度、抖音开放平台第三方代小程序开发总结
大家好,我是小悟 小伙伴们都开启小长假了吧,值此中秋国庆双节之际,小悟祝所有的小伙伴们节日快乐。 支付宝社区很用心,还特意给寄了袋月饼,愿中秋节的圆月带给你身体健康,幸福团圆,国庆节的旗帜…...
C语言协程
协程(Coroutine)是一种程序运行方式,相比于线程和进程,协程更加轻量级,可以被视为一种用户态的线程,不需要内核的参与。 协程的特点在于其执行过程中可以被挂起(Suspend)࿰…...

stm32G473的flash模式是单bank还是双bank?
今天突然有人stm32G473的flash模式是单bank还是双bank?由于时间太久,我真忘记了。搜搜发现,还真有人和我一样。见下面的链接:https://shequ.stmicroelectronics.cn/forum.php?modviewthread&tid644563 根据STM32G4系列参考手…...
Python爬虫实战:研究feedparser库相关技术
1. 引言 1.1 研究背景与意义 在当今信息爆炸的时代,互联网上存在着海量的信息资源。RSS(Really Simple Syndication)作为一种标准化的信息聚合技术,被广泛用于网站内容的发布和订阅。通过 RSS,用户可以方便地获取网站更新的内容,而无需频繁访问各个网站。 然而,互联网…...
1688商品列表API与其他数据源的对接思路
将1688商品列表API与其他数据源对接时,需结合业务场景设计数据流转链路,重点关注数据格式兼容性、接口调用频率控制及数据一致性维护。以下是具体对接思路及关键技术点: 一、核心对接场景与目标 商品数据同步 场景:将1688商品信息…...

微服务商城-商品微服务
数据表 CREATE TABLE product (id bigint(20) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT 商品id,cateid smallint(6) UNSIGNED NOT NULL DEFAULT 0 COMMENT 类别Id,name varchar(100) NOT NULL DEFAULT COMMENT 商品名称,subtitle varchar(200) NOT NULL DEFAULT COMMENT 商…...

vue3+vite项目中使用.env文件环境变量方法
vue3vite项目中使用.env文件环境变量方法 .env文件作用命名规则常用的配置项示例使用方法注意事项在vite.config.js文件中读取环境变量方法 .env文件作用 .env 文件用于定义环境变量,这些变量可以在项目中通过 import.meta.env 进行访问。Vite 会自动加载这些环境变…...

selenium学习实战【Python爬虫】
selenium学习实战【Python爬虫】 文章目录 selenium学习实战【Python爬虫】一、声明二、学习目标三、安装依赖3.1 安装selenium库3.2 安装浏览器驱动3.2.1 查看Edge版本3.2.2 驱动安装 四、代码讲解4.1 配置浏览器4.2 加载更多4.3 寻找内容4.4 完整代码 五、报告文件爬取5.1 提…...

MFC 抛体运动模拟:常见问题解决与界面美化
在 MFC 中开发抛体运动模拟程序时,我们常遇到 轨迹残留、无效刷新、视觉单调、物理逻辑瑕疵 等问题。本文将针对这些痛点,详细解析原因并提供解决方案,同时兼顾界面美化,让模拟效果更专业、更高效。 问题一:历史轨迹与小球残影残留 现象 小球运动后,历史位置的 “残影”…...

c++第七天 继承与派生2
这一篇文章主要内容是 派生类构造函数与析构函数 在派生类中重写基类成员 以及多继承 第一部分:派生类构造函数与析构函数 当创建一个派生类对象时,基类成员是如何初始化的? 1.当派生类对象创建的时候,基类成员的初始化顺序 …...

Linux部署私有文件管理系统MinIO
最近需要用到一个文件管理服务,但是又不想花钱,所以就想着自己搭建一个,刚好我们用的一个开源框架已经集成了MinIO,所以就选了这个 我这边对文件服务性能要求不是太高,单机版就可以 安装非常简单,几个命令就…...

协议转换利器,profinet转ethercat网关的两大派系,各有千秋
随着工业以太网的发展,其高效、便捷、协议开放、易于冗余等诸多优点,被越来越多的工业现场所采用。西门子SIMATIC S7-1200/1500系列PLC集成有Profinet接口,具有实时性、开放性,使用TCP/IP和IT标准,符合基于工业以太网的…...