0

0

C++中如何拦截系统信号_C++中signal函数处理Ctrl+C退出【系统】

穿越時空

穿越時空

发布时间:2026-02-07 14:24:12

|

534人浏览过

|

来源于php中文网

原创

不能可靠拦截 Ctrl+C,因 signal() 非线程安全、行为不一致、多次触发易崩溃,且被库调用干扰;应使用 sigaction() 原子注册并配合 volatile sig_atomic_t 标志位延迟清理。

c++中如何拦截系统信号_c++中signal函数处理ctrl+c退出【系统】

signal() 能否可靠拦截 Ctrl+C

signal() 可以注册对 SIGINT 的处理函数,从而响应 Ctrl+C,但**它不是线程安全的,且行为在不同系统上不一致(尤其是信号被多次触发时)**。POSIX 明确指出 signal() 是为兼容旧代码保留的接口,sigaction() 才是推荐方案。

常见错误现象:
- 注册后按 Ctrl+C 无反应(可能被后续库调用覆盖)
- 多次 Ctrl+C 导致程序崩溃(signal() 默认重置为 SIG_DFL
- 在多线程环境中信号被任意线程接收,无法控制

  • 使用 signal(SIGINT, handler) 后,一旦触发一次,部分系统会自动恢复为默认行为,需手动再次调用 signal()(不推荐)
  • 若程序中用了 std::thread 或第三方库(如 Boost、Qt),它们可能内部调用 sigprocmaskpthread_sigmask,干扰 signal() 效果
  • SIGINT 是唯一保证由终端发送给前台进程组的信号,但仅当进程未忽略或阻塞它时才生效

用 sigaction() 正确注册 SIGINT 处理器

sigaction() 提供原子性注册、可屏蔽其他信号、支持重启被中断的系统调用等能力,是 C++ 中拦截 Ctrl+C 的实际标准做法。

关键点:
- 必须显式设置 sa_flags(例如 SA_RESTART 避免 read() 等调用被中断)
- sa_mask 可临时阻塞其他信号,防止嵌套调用(如 SIGINT 处理中又收到 SIGUSR1
- 不要从信号处理器中调用非异步信号安全函数(如 std::coutmallocprintf

struct sigaction sa;
sa.sa_handler = [](int) {
    // 仅允许异步信号安全操作:写入全局 volatile 变量、_exit()、siglongjmp()
    g_should_exit = true;  // 声明为 volatile sig_atomic_t
};
sigemptyset(&sa.sa_mask);
sa.sa_flags = SA_RESTART;  // 让阻塞系统调用自动重试
sigaction(SIGINT, &sa, nullptr);

volatile sig_atomic_t 是什么,为什么不能用 bool

信号处理器可能在任意指令中间打断主流程,普通 bool 变量的读写不是原子的,编译器还可能对其做优化(比如缓存到寄存器),导致主循环永远看不到变化。

  • sig_atomic_t 是 C/C++ 标准定义的、能被信号处理器安全读写的整数类型(通常为 int
  • 必须配合 volatile 使用,禁止编译器优化掉重复读取
  • 声明示例:volatile sig_atomic_t g_should_exit = 0;
  • 主循环中应这样检查:while (!g_should_exit) { ... },而非 while (true) + 内部 break

退出前清理资源要注意什么

信号处理器里不能调用 deleteclose()fclose()std::mutex::unlock() 等——这些都不是异步信号安全函数。

影谱
影谱

汉语电影AI辅助创作平台

下载

立即学习C++免费学习笔记(深入)”;

正确做法是「延迟处理」:
- 信号处理器只设标志位
- 主线程检测到标志后,按常规逻辑执行清理(关闭文件、释放内存、join 线程等)
- 若使用 std::thread,确保所有线程已 join 或 detach,否则 main() 返回会终止整个进程,未 join 的线程被强制销毁(C++11 起会调用 std::terminate

容易被忽略的一点:某些系统调用(如 epoll_waitselect)在被 SIGINT 中断后返回 -1 并设 errno = EINTR,需主动重试或退出,否则程序可能卡住不动。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
视频后缀名都有哪些
视频后缀名都有哪些

视频后缀名都有avi、mpg、mpeg、rm、rmvb、flv、wmv、mov、mkv、ASF、M1V、M2V、MPE、QT、VOB、RA、RMJ、RMS、RAM、等等。更多关于视频后缀名的相关知识,详情请看本专题下面的文章,php中文网欢迎大家前来学习。

3585

2023.10.31

C++ Qt图形开发
C++ Qt图形开发

本专题专注于 C++ Qt框架在图形界面开发中的应用,系统讲解窗口设计、信号与槽机制、界面布局、事件处理、数据库连接与跨平台打包等核心技能,通过多个桌面应用项目实战,帮助学员快速掌握 Qt 框架并独立完成跨平台GUI软件的开发。

71

2025.08.15

C++ 图形界面开发基础(Qt方向)
C++ 图形界面开发基础(Qt方向)

本专题系统讲解 使用 C++ 与 Qt 进行图形界面(GUI)开发的核心技能,内容涵盖 Qt 项目结构、窗口组件、信号与槽机制、事件处理、布局管理、资源管理,以及跨平台编译与打包流程。通过多个小型桌面应用实战案例,帮助学习者掌握从界面设计到功能实现的完整 GUI 开发能力。

82

2025.12.05

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

98

2023.09.25

fclose函数的用法
fclose函数的用法

fclose是一个C语言和C++中的标准库函数,用于关闭一个已经打开的文件,是文件操作中非常重要的一个函数,用于将文件流与底层文件系统分离,释放相关的资源。更多关于fclose函数的相关问题,详情请看本专题下面的文章。php中文网欢迎大家前来学习。

334

2023.11.30

printf用法大全
printf用法大全

php中文网为大家提供printf用法大全,以及其他printf函数的相关文章、相关下载资源以及各种相关课程,供大家免费下载体验。

75

2023.06.20

fprintf和printf的区别
fprintf和printf的区别

fprintf和printf的区别在于输出的目标不同,printf输出到标准输出流,而fprintf输出到指定的文件流。根据需要选择合适的函数来进行输出操作。更多关于fprintf和printf的相关文章详情请看本专题下面的文章。php中文网欢迎大家前来学习。

288

2023.11.28

java中break的作用
java中break的作用

本专题整合了java中break的用法教程,阅读专题下面的文章了解更多详细内容。

120

2025.10.15

Golang处理数据库错误教程合集
Golang处理数据库错误教程合集

本专题整合了Golang数据库错误处理方法、技巧、管理策略相关内容,阅读专题下面的文章了解更多详细内容。

2

2026.02.06

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Django 教程
Django 教程

共28课时 | 4万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.4万人学习

Sass 教程
Sass 教程

共14课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号