C++通过系统API实现Unix/Linux信号处理,使用signal()或更安全的sigaction()注册处理函数,捕获如SIGINT、SIGTERM等信号,响应中断或终止请求。示例显示按下Ctrl+C触发SIGINT,调用自定义处理函数输出信息并退出;另一示例用sigaction处理SIGTERM,支持可靠信号管理。注意事项包括:仅在信号处理函数中调用异步信号安全函数,避免复杂操作,推荐设置volatile sig_atomic_t标志位,主循环中检查并响应,确保线程与异常安全。合理使用可实现程序优雅关闭与外部命令响应。

在Unix/Linux系统中,信号(Signal)是一种用于进程间通信的机制,常用于通知进程发生了某种事件,比如用户按下Ctrl+C、程序出现段错误、定时器超时等。C++本身没有内置的信号处理机制,但可以借助操作系统提供的API来实现信号的捕获和响应。下面介绍C++在Linux环境下如何处理信号。
信号的基本概念
信号是软件中断,由内核或进程发送给另一个进程,表示某个特殊事件的发生。每个信号都有一个唯一的编号和默认行为,例如终止进程、暂停执行或忽略。常见的信号包括:- SIGINT:用户按下Ctrl+C,通常用于中断程序
- SIGTERM:请求进程终止,可被捕获和处理
- SIGKILL:强制终止进程,不能被捕获或忽略
- SIGSEGV:段错误,访问非法内存时触发
- SIGALRM:定时器超时信号
使用signal()函数注册信号处理器
最简单的方式是使用signal()函数设置信号处理函数。该函数定义在头文件中。
示例:捕获SIGINT信号(Ctrl+C)
#include运行程序后,按下Ctrl+C会触发SIGINT,调用#include #include void signalHandler(int sig) { std::cout << "\n接收到信号 " << sig << ",正在退出...\n"; exit(sig); } int main() { // 注册SIGINT信号处理函数 signal(SIGINT, signalHandler); std::cout << "等待信号(按 Ctrl+C 中断)...\n"; while (true) { // 模拟工作 } return 0; }
signalHandler函数而不是直接终止程序。
使用sigaction进行更安全的信号处理
相比signal(),sigaction提供了更精确和可靠的控制,推荐在生产环境中使用。
示例:使用sigaction处理SIGTERM
#include#include #include void signalHandler(int sig) { if (sig == SIGTERM) { std::cout << "收到终止请求,清理资源并退出。\n"; } exit(0); } int main() { struct sigaction sa; std::memset(&sa, 0, sizeof(sa)); sa.sa_handler = signalHandler; sigemptyset(&sa.sa_mask); // 不阻塞其他信号 sa.sa_flags = 0; if (sigaction(SIGTERM, &sa, nullptr) == -1) { perror("sigaction"); return 1; } std::cout << "程序运行中,尝试使用 'kill ' 发送SIGTERM\n"; while (true) {} return 0; }
sigaction的优点包括:
- 避免信号处理函数被重置为默认行为
- 可指定阻塞哪些信号在处理期间不被接收
- 支持更多标志位控制行为
信号处理中的注意事项
虽然信号处理很强大,但在C++中使用时需注意以下几点:- 信号处理函数中只能调用异步信号安全函数(如
write、_exit),不能使用std::cout、malloc、printf等非安全函数 - 避免在信号处理中做复杂操作,建议只设置标志位,主循环中检查并响应
- 不要在信号处理函数中抛出异常或调用C++异常机制,行为未定义
- 某些信号如SIGSEGV不能安全恢复,处理后应尽快退出
推荐做法:在信号处理函数中仅修改一个volatile sig_atomic_t变量,主程序循环中检查该变量。
立即学习“C++免费学习笔记(深入)”;
volatile sig_atomic_t signalReceived = 0;
void signalHandler(int sig) {
signalReceived = sig; // 安全赋值
}
int main() {
signal(SIGINT, signalHandler);
while (!signalReceived) {
// 正常工作
}
std::cout << "程序被信号 " << signalReceived << " 中断\n";
return 0;
}
基本上就这些。C++通过系统调用支持信号处理,关键在于理解信号的异步性和安全限制。合理使用能提升程序健壮性,比如优雅关闭服务、响应外部控制命令等。











