C++怎么处理信号(Signal)_C++在Unix/Linux环境下的信号处理与编程

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 
#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;
}
运行程序后,按下Ctrl+C会触发SIGINT,调用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::coutmallocprintf等非安全函数
  • 避免在信号处理中做复杂操作,建议只设置标志位,主循环中检查并响应
  • 不要在信号处理函数中抛出异常或调用C++异常机制,行为未定义
  • 某些信号如SIGSEGV不能安全恢复,处理后应尽快退出

推荐做法:在信号处理函数中仅修改一个volatile sig_atomic_t变量,主程序循环中检查该变量。

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++通过系统调用支持信号处理,关键在于理解信号的异步性和安全限制。合理使用能提升程序健壮性,比如优雅关闭服务、响应外部控制命令等。