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,结果进程直接终止。

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

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

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

下载

解决方案是:在 所有线程创建前,用 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(),否则主线程退出时未结束的线程会引发未定义行为

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

相关专题

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

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

72

2023.06.20

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

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

280

2023.11.28

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

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

67

2025.10.23

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

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

389

2023.07.18

堆和栈区别
堆和栈区别

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

571

2023.08.10

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

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

389

2023.07.18

堆和栈区别
堆和栈区别

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

571

2023.08.10

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

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

480

2023.08.10

Golang gRPC 服务开发与Protobuf实战
Golang gRPC 服务开发与Protobuf实战

本专题系统讲解 Golang 在 gRPC 服务开发中的完整实践,涵盖 Protobuf 定义与代码生成、gRPC 服务端与客户端实现、流式 RPC(Unary/Server/Client/Bidirectional)、错误处理、拦截器、中间件以及与 HTTP/REST 的对接方案。通过实际案例,帮助学习者掌握 使用 Go 构建高性能、强类型、可扩展的 RPC 服务体系,适用于微服务与内部系统通信场景。

8

2026.01.15

热门下载

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

精品课程

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

共48课时 | 7.2万人学习

Git 教程
Git 教程

共21课时 | 2.7万人学习

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

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