0

0

c++中如何处理SIGTERM信号以实现程序的优雅退出? (信号处理)

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-15 16:06:09

|

377人浏览过

|

来源于php中文网

原创

不能直接用 signal() 注册 sigterm 处理函数,因其行为跨平台不一致、默认重置处理方式且不保证异步信号安全;应使用 sigaction() 配合 volatile std::sig_atomic_t 标志和主循环检查实现安全退出。

c++中如何处理sigterm信号以实现程序的优雅退出? (信号处理)

为什么不能直接用 signal() 注册 SIGTERM 处理函数?

因为 signal() 行为在不同系统上不一致(POSIX vs. BSD),且默认会重置信号处理方式,导致第二次 SIGTERM 可能被忽略或触发默认终止。更严重的是,signal() 不保证信号安全——在处理函数中调用 printfstd::coutnew 等非异步信号安全函数会引发未定义行为。

必须改用 sigaction(),它可精确控制信号掩码、标志位,并保证可重入性。

如何用 sigaction 安全注册 SIGTERM 处理器

核心是:只做最少必要操作(设标志位),其余清理逻辑放在主循环中检查。以下是最小可靠模板:

volatile std::sig_atomic_t g_terminate_requested = 0;

void sigterm_handler(int sig) {
    if (sig == SIGTERM) {
        g_terminate_requested = 1;  // 只写 sig_atomic_t,异步信号安全
    }
}

int main() {
    struct sigaction sa;
    sa.sa_handler = sigterm_handler;
    sigemptyset(&sa.sa_mask);      // 不额外屏蔽其他信号
    sa.sa_flags = SA_RESTART;      // 让被中断的系统调用自动重启
    sigaction(SIGTERM, &sa, nullptr);

    while (!g_terminate_requested) {
        // 主工作循环(如网络收发、定时任务)
        // 注意:sleep()/poll()/epoll_wait() 等可被信号中断,需检查返回值
        poll(nullptr, 0, 1000);  // 示例:每秒检查一次退出标志
    }

    // 此处执行所有非信号安全的清理:关闭文件、释放内存、日志落盘等
    cleanup_resources();
    return 0;
}
  • g_terminate_requested 必须是 volatile std::sig_atomic_t 类型,确保写入不会被编译器优化掉,且是原子读写
  • SA_RESTART 很关键:否则 poll()read() 等可能返回 -1 并设 errno = EINTR,需手动重试
  • 绝对不要在 sigterm_handler 里调用 std::coutfree()pthread_cancel() 等非 async-signal-safe 函数

多线程环境下 SIGTERM 怎么投递给正确线程?

Linux 中信号默认投递给进程的任意一个未屏蔽该信号的线程。如果主线程屏蔽了 SIGTERM,而工作线程没屏蔽,就可能由工作线程接收——但工作线程通常没注册 handler,结果进程直接终止。

动态WEB网站中的PHP和MySQL:直观的QuickPro指南第2版
动态WEB网站中的PHP和MySQL:直观的QuickPro指南第2版

动态WEB网站中的PHP和MySQL详细反映实际程序的需求,仔细地探讨外部数据的验证(例如信用卡卡号的格式)、用户登录以及如何使用模板建立网页的标准外观。动态WEB网站中的PHP和MySQL的内容不仅仅是这些。书中还提到如何串联JavaScript与PHP让用户操作时更快、更方便。还有正确处理用户输入错误的方法,让网站看起来更专业。另外还引入大量来自PEAR外挂函数库的强大功能,对常用的、强大的包

下载

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

解决方案是:在 所有线程创建前,用 pthread_sigmask() 屏蔽 SIGTERM,然后在主线程中用 sigwait()signalfd()(Linux 特有)同步等待:

// 主线程初始化时:
sigset_t set;
sigemptyset(&set);
sigaddset(&set, SIGTERM);
pthread_sigmask(SIG_BLOCK, &set, nullptr);  // 阻塞 SIGTERM

// 后续在主线程循环中:
int sig;
sigwait(&set, &sig);  // 同步等待,安全,可调用任意函数
if (sig == SIGTERM) {
    g_terminate_requested = 1;
}
  • 必须在 pthread_create() 前调用 pthread_sigmask(),否则子线程会继承默认信号掩码
  • sigwait() 是唯一被 POSIX 明确标记为 async-signal-safe 的信号等待方式
  • 若用 signalfd(),需注意它返回的是文件描述符,可与 epoll 集成,但仅限 Linux

常见陷阱:程序仍被立即杀死,根本没走 cleanup

最常被忽略的是:某些第三方库(如 glibc 的 malloc、日志库)内部可能调用 abort() 或触发 SIGABRT;或者程序在收到 SIGTERM 后,又因 bug 收到 SIGSEGV,导致直接崩溃。

  • 务必用 sigaction()SIGABRTSIGSEGVSIGILL 等致命信号也设置 handler,至少记录堆再退出
  • 避免在信号 handler 中修改全局对象(如 std::vector),哪怕只读——其内部可能调用非信号安全函数
  • 若使用 std::thread,确保所有线程已 join()detach(),否则主线程退出时未结束的线程会引发未定义行为

优雅退出的关键不在“怎么捕获信号”,而在“捕获后如何让整个程序状态可控地收敛”。信号只是中断点,真正的退出逻辑必须在主控制流中完成。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
printf用法大全
printf用法大全

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

76

2023.06.20

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

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

303

2023.11.28

c++中volatile关键字的作用
c++中volatile关键字的作用

本专题整合了c++中volatile关键字的相关内容,阅读专题下面的文章了解更多详细内容。

75

2025.10.23

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

435

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

601

2023.08.10

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

435

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

601

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

763

2023.08.10

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

23

2026.03.06

热门下载

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

精品课程

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

共48课时 | 10.3万人学习

Git 教程
Git 教程

共21课时 | 4.1万人学习

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

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