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

学懂C++ (十九):高级教程——深入详解C++信号处理

目录

C++中的信号处理

1. 信号处理的本质

2. 主要信号类型

3. 核心关键点

4. 经典实例

代码分析

5. 进阶:信号屏蔽与多线程

例子:使用sigaction()

6. Windows中的信号处理

7. 比较与总结

示例:Windows控制台事件处理

总结


C++中的信号处理

C++中的信号处理主要涉及操作系统层面的信号机制,尤其是在Unix和Linux系统中。信号是一种异步事件的通知机制,允许程序在特定事件发生时进行响应。常见的用途包括处理异常情况、外部中断等。以下将深入探讨信号处理的本质、信号类型、核心关键点以及经典实例。

1. 信号处理的本质

信号处理的本质在于允许程序响应异步事件。当特定事件发生时,操作系统向进程发送信号,进程通过信号处理程序(信号处理函数)来响应这些信号。信号的用途包括:

  • 处理用户请求(如Ctrl+C中断)
  • 处理定时器
  • 响应硬件异常
  • 实现进程间通信

2. 主要信号类型

在Unix/Linux系统中,常见的信号包括:

  • SIGINT:中断信号(通常由Ctrl+C触发)
  • SIGTERM:终止信号(请求程序终止)
  • SIGSEGV:段错误(访问无效内存)
  • SIGALRM:定时器到期

3. 核心关键点

在进行信号处理时,需要掌握以下核心要点:

  • 信号的注册与处理:可以使用signal()sigaction()函数来注册信号处理程序。
  • 信号阻塞与解除阻塞:可以用sigprocmask()控制信号的阻塞,以防在特定代码段内被处理。
  • 信号发送:可以使用kill()函数向进程发送信号。
  • 并发性:信号处理是异步的,可能与程序的其他部分并发执行,因此需要注意线程安全。

4. 经典实例

以下是一个简单的信号处理示例,演示如何处理SIGINT信号(通常由Ctrl+C触发)。

#include <iostream>
#include <csignal>
#include <unistd.h> // for sleep()// 信号处理函数
void signalHandler(int signum) {std::cout << "Caught signal " << signum << ", terminating gracefully..." << std::endl;// 这里可以做一些清理工作exit(signum); // 退出程序
}int main() {// 注册信号处理程序signal(SIGINT, signalHandler);std::cout << "Press Ctrl+C to trigger the signal handler..." << std::endl;// 程序主循环while (true) {std::cout << "Running..." << std::endl;sleep(1); // 每秒打印一次}return 0;
}
代码分析
  1. 注册信号处理程序:使用signal(SIGINT, signalHandler)注册信号处理函数。当程序接收到SIGINT信号时,调用signalHandler函数。

  2. 信号处理逻辑:在signalHandler中进行清理工作并输出通知,最后调用exit(signum)退出程序。

  3. 主循环:程序在无限循环中持续运行,每秒打印一次“Running...”。按下Ctrl+C时,将捕获到SIGINT信号,调用处理程序并优雅地终止。

5. 进阶:信号屏蔽与多线程

在多线程程序中,信号处理变得更加复杂。关键要点包括:

  • 信号屏蔽:使用sigprocmask()阻塞某些信号,以防在特定代码块内被处理。
  • 专用信号线程:创建专门的线程处理信号,避免信号处理函数与其他线程并发执行的问题。
  • 使用sigaction()sigaction()提供丰富的功能,允许设置信号处理的行为,如恢复默认处理和设置信号屏蔽字。
例子:使用sigaction()
#include <iostream>
#include <csignal>
#include <cstring>
#include <unistd.h>// 信号处理函数
void signalHandler(int signum, siginfo_t *info, void *context) {std::cout << "Caught signal " << signum << ", terminating gracefully..." << std::endl;// 这里可以做一些清理工作exit(signum); // 退出程序
}int main() {struct sigaction action;memset(&action, 0, sizeof(action));action.sa_sigaction = signalHandler; // 设置处理函数action.sa_flags = SA_SIGINFO; // 使用siginfo_t结构// 注册信号处理程序if (sigaction(SIGINT, &action, nullptr) == -1) {std::cerr << "Error registering signal handler" << std::endl;return 1;}std::cout << "Press Ctrl+C to trigger the signal handler..." << std::endl;// 程序主循环while (true) {std::cout << "Running..." << std::endl;sleep(1); // 每秒打印一次}return 0;
}

6. Windows中的信号处理

在Windows系统中,信号的概念与Unix/Linux系统有所不同。Windows使用以下机制处理异步事件:

  • 异常处理:使用结构化异常处理(SEH)来处理运行时错误,允许捕获异常并执行处理逻辑。
  • 控制台控制处理:通过SetConsoleCtrlHandler()函数注册控制台控制处理程序,响应如Ctrl+C等事件。

7. 比较与总结

  • 信号机制:Unix/Linux提供标准的信号机制,允许程序使用信号进行异步事件处理;而Windows则通过异常处理和控制台事件机制完成类似功能。
  • 跨平台性:虽然C++标准库提供了一些跨平台功能,但信号处理在不同操作系统中的实现方式和可用API不尽相同。因此,编写跨平台的信号处理代码时,需要考虑不同操作系统的特性和差异。

示例:Windows控制台事件处理

#include <windows.h>
#include <iostream>BOOL WINAPI ConsoleHandler(DWORD signal) {if (signal == CTRL_C_EVENT) {std::cout << "Caught Ctrl+C! Exiting gracefully..." << std::endl;return TRUE; // 阻止默认处理(程序退出)}return FALSE; // 让系统执行默认处理
}int main() {// 注册控制台控制处理程序SetConsoleCtrlHandler(ConsoleHandler, TRUE);std::cout << "Press Ctrl+C to trigger the handler..." << std::endl;// 程序主循环while (true) {std::cout << "Running..." << std::endl;Sleep(1000); // 每秒打印一次}return 0;
}

总结

C++中的信号处理是一个重要的机制,能够让程序对异步事件做出反应。理解信号的基本概念、如何注册处理程序、信号的阻塞与解除以及多线程环境中的处理逻辑是掌握信号处理的核心。尽管在不同操作系统下信号处理机制存在差异,了解这些差异有助于编写更加健壮和可移植的代码。

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • 初识C++ · C++11(2)
  • 若依 ruoyi 单体双token(url区分)
  • Linux 软件编程学习第十一天
  • 使用RPC服务的步骤
  • python打怪练习
  • UEFI ——Firmware层级结构
  • [数据集][目标检测]轴承缺陷划痕检测数据集VOC+YOLO格式1166张1类别
  • wordpress评论ip异常问题
  • 美团面经到店研发
  • 微服务的多面手:Spring Cloud 多数据中心支持全解析
  • 使用Python+moviepy保存截取视频画面
  • javaweb_07:分层解耦
  • Java之TCP编程综合案例
  • 卷积神经网络的相关知识点
  • C++相关概念和易错语法(25)(列表初始化、initializer_list)
  • Android开源项目规范总结
  • Bootstrap JS插件Alert源码分析
  • CAP 一致性协议及应用解析
  • CSS3 聊天气泡框以及 inherit、currentColor 关键字
  • ES6, React, Redux, Webpack写的一个爬 GitHub 的网页
  • gitlab-ci配置详解(一)
  • HTTP那些事
  • JavaScript对象详解
  • JS实现简单的MVC模式开发小游戏
  • pdf文件如何在线转换为jpg图片
  • Python进阶细节
  • vue2.0项目引入element-ui
  • 聊聊flink的BlobWriter
  • 聊聊spring cloud的LoadBalancerAutoConfiguration
  • 设计模式 开闭原则
  • 限制Java线程池运行线程以及等待线程数量的策略
  • 移动端解决方案学习记录
  • 1.Ext JS 建立web开发工程
  • 智能情侣枕Pillow Talk,倾听彼此的心跳
  • # 计算机视觉入门
  • #NOIP 2014# day.2 T2 寻找道路
  • #进阶:轻量级ORM框架Dapper的使用教程与原理详解
  • $con= MySQL有关填空题_2015年计算机二级考试《MySQL》提高练习题(10)
  • ()、[]、{}、(())、[[]]等各种括号的使用
  • (0)Nginx 功能特性
  • (13)Hive调优——动态分区导致的小文件问题
  • (3)医疗图像处理:MRI磁共振成像-快速采集--(杨正汉)
  • (30)数组元素和与数字和的绝对差
  • (ros//EnvironmentVariables)ros环境变量
  • (void) (_x == _y)的作用
  • (二十一)devops持续集成开发——使用jenkins的Docker Pipeline插件完成docker项目的pipeline流水线发布
  • (翻译)Quartz官方教程——第一课:Quartz入门
  • (附表设计)不是我吹!超级全面的权限系统设计方案面世了
  • (附源码)ssm码农论坛 毕业设计 231126
  • (函数)颠倒字符串顺序(C语言)
  • (十六)视图变换 正交投影 透视投影
  • (算法设计与分析)第一章算法概述-习题
  • (转)原始图像数据和PDF中的图像数据
  • (转载)OpenStack Hacker养成指南
  • .NET Micro Framework 4.2 beta 源码探析