C++中处理信号依赖操作系统机制,通过signal或sigaction注册处理函数响应事件。1. 使用signal函数可捕获如SIGINT(Ctrl+C)等信号,执行自定义逻辑。2. 更推荐使用sigaction,因其具备更好可移植性与控制能力。3. 信号处理函数应仅调用异步信号安全函数,通常设置全局标志位通知主循环退出。4. 常见实践是在处理函数中修改volatile sig_atomic_t类型标志,主循环检测后安全终止程序。掌握信号处理对编写健壮服务程序至关重要。

在C++中处理信号主要依赖于操作系统提供的信号机制,尤其是类Unix系统(如Linux、macOS)中的signal和sigaction函数。信号是一种软件中断,用于通知进程发生了某种事件,比如用户按下Ctrl+C(触发SIGINT)、程序访问非法内存(触发SIGSEGV)等。
信号处理的基本方式
C++本身不提供原生的信号处理语法,而是通过调用C标准库中的signal()函数来注册信号处理器(也叫信号捕捉函数)。当特定信号到达时,程序会中断当前执行流程,转而执行预先设定的处理函数。
常用信号包括:
- SIGINT:用户按下Ctrl+C时触发,默认行为是终止程序
- SIGTERM:请求终止进程,可被捕获
- SIGKILL:强制终止进程,不能被捕获或忽略
- SIGSEGV:段错误,访问非法内存时触发
使用signal()注册SIGINT处理函数
下面是一个简单的示例,展示如何捕获Ctrl+C(即SIGINT信号),并自定义其行为:
立即学习“C++免费学习笔记(深入)”;
#include#include #include // 信号处理函数 void signalHandler(int signum) { std::cout << "\n接收到信号 " << signum << ",正在退出...\n"; exit(signum); // 正常退出程序 } int main() { // 注册SIGINT信号的处理函数 std::signal(SIGINT, signalHandler); std::cout << "等待Ctrl+C...\n"; // 模拟长时间运行的任务 while (true) { // 可以加入实际工作逻辑 } return 0; }
当你运行这个程序并按下Ctrl+C时,不会立即退出,而是跳转到signalHandler函数,打印提示信息后再退出。
更安全的方式:使用sigaction
虽然signal()简单易用,但在某些系统上行为不一致。推荐使用sigaction进行更精确的控制,它可以避免一些潜在问题,比如信号处理期间是否重置处理函数。
#include#include #include void signalHandler(int signum) { std::cout << "\n捕获到信号: " << signum << "\n"; exit(signum); } int main() { struct sigaction sa; sa.sa_handler = signalHandler; sigemptyset(&sa.sa_mask); sa.sa_flags = 0; // 设置SIGINT的处理动作 sigaction(SIGINT, &sa, nullptr); std::cout << "等待信号 (按 Ctrl+C 中断)...\n"; while (true) { } return 0; }
sigaction提供了更强的可移植性和控制能力,特别是在多线程或复杂应用中更推荐使用。
注意事项与限制
信号处理函数有严格限制,并非所有操作都能安全执行。只能调用“异步信号安全”(async-signal-safe)的函数,例如write、_exit,而不能使用cout、malloc、printf等(尽管某些实现允许,但不可靠)。
因此,在实际项目中,常见的做法是在信号处理函数中仅修改一个全局标志位,主循环检测该标志后决定是否退出:
#include#include volatile sig_atomic_t stop = 0; void signalHandler(int signum) { stop = 1; // 仅设置标志,避免复杂操作 } int main() { std::signal(SIGINT, signalHandler); while (!stop) { // 执行主任务 std::cout << "运行中... (按 Ctrl+C 停止)\r" << std::flush; usleep(500000); // 半秒 } std::cout << "\n程序已安全退出。\n"; return 0; }
这种方式既响应了信号,又保证了程序逻辑的安全性。
基本上就这些。C++信号处理虽源自C,但在服务程序、守护进程或需要优雅关闭的应用中非常实用。掌握SIGINT这类基础信号的处理,是编写健壮系统的必要技能。










