C++ csignal库详细使用介绍

C++ 的 <csignal> 库提供了一种处理操作系统信号的机制,允许程序响应外部事件(如用户中断或程序错误)。它是从 C 语言的 <signal.h> 继承而来,属于 C++ 标准库的一部分,用于跨平台的基本信号处理。以下是详细介绍:


1. 核心功能

  • 信号(Signal) :软件中断,由操作系统或程序自身触发的事件(例如 Ctrl+C 生成 SIGINT)。
  • 信号处理函数(Signal Handler):用户定义的函数,用于响应特定信号。
  • 发送信号 :程序可以通过 raise() 主动触发信号。

2. 主要组件

2.1 信号宏

<csignal> 定义了以下常见信号宏:

信号 描述
SIGABRT 程序异常终止(如 abort() 调用)
SIGFPE 算术运算错误(如除以零)
SIGILL 非法指令(如执行无效的机器指令)
SIGINT 交互式中断(如 Ctrl+C
SIGSEGV 无效内存访问(如解引用空指针)
SIGTERM 终止请求(如 kill 命令)
2.2 函数
  • signal(int sig, void (*handler)(int))

    注册信号处理函数。参数为信号编号和处理函数指针。

    示例:signal(SIGINT, handler_func);

  • int raise(int sig)

    向当前程序发送信号 sig。成功返回 0,失败返回非零。


3. 基本用法示例

cpp 复制代码
#include <csignal>
#include <iostream>
#include <unistd.h> // 用于 sleep()

volatile sig_atomic_t flag = 0;

void signal_handler(int sig) {
    if (sig == SIGINT) {
        std::cout << "\nReceived SIGINT. Exiting gracefully...\n";
        flag = 1;
    }
}

int main() {
    // 注册信号处理函数
    signal(SIGINT, signal_handler);

    std::cout << "Running. Press Ctrl+C to exit.\n";
    while (!flag) {
        sleep(1); // 模拟程序运行
    }
    return 0;
}

输出示例:

复制代码
Running. Press Ctrl+C to exit.
^C
Received SIGINT. Exiting gracefully...

4. 注意事项

4.1 异步安全性(Async-Safety)

信号处理函数可能在任何时间点中断主程序,因此内部只能使用 异步安全函数 (如 write()),避免使用 coutmalloc 等非安全操作。

4.2 volatile sig_atomic_t

在信号处理函数和主程序之间共享变量时,应使用 volatile sig_atomic_t 类型,确保原子性和可见性。

4.3 可移植性
  • signal() 的行为因系统而异(如 Windows 和 Unix-like 系统差异)。
  • 推荐使用 POSIX 的 sigaction(需 <signal.h>)替代 signal(),以获得更可靠的控制。
4.4 多线程环境

信号处理在多线程中行为复杂,通常建议在主线程统一处理信号,并避免在信号处理中操作共享资源。


5. 替代方案

  • POSIX sigaction:提供更精细的信号控制(如阻塞信号、指定标志位)。
  • C++ 异常:无法直接替代信号处理,但可通过将信号转换为异常(需谨慎)。
  • Boost.Signals2:基于事件的信号/槽机制,适合高层应用逻辑。

6. 总结

<csignal> 是 C++ 中处理底层信号的轻量级工具,适合简单场景。对于复杂需求,建议结合系统特定的 API(如 sigaction)或高级库。始终注意异步安全性和线程安全问题,避免不可预测的行为。

相关推荐
LawrenceLan4 小时前
Flutter 零基础入门(十一):空安全(Null Safety)基础
开发语言·flutter·dart
txinyu的博客4 小时前
解析业务层的key冲突问题
开发语言·c++·分布式
码不停蹄Zzz5 小时前
C语言第1章
c语言·开发语言
行者965 小时前
Flutter跨平台开发在OpenHarmony上的评分组件实现与优化
开发语言·flutter·harmonyos·鸿蒙
阿蒙Amon5 小时前
C#每日面试题-Array和ArrayList的区别
java·开发语言·c#
SmartRadio6 小时前
ESP32添加修改蓝牙名称和获取蓝牙连接状态的AT命令-完整UART BLE服务功能后的完整`main.c`代码
c语言·开发语言·c++·esp32·ble
且去填词6 小时前
Go 语言的“反叛”——为什么少即是多?
开发语言·后端·面试·go
知乎的哥廷根数学学派6 小时前
基于生成对抗U-Net混合架构的隧道衬砌缺陷地质雷达数据智能反演与成像方法(以模拟信号为例,Pytorch)
开发语言·人工智能·pytorch·python·深度学习·机器学习
yeziyfx7 小时前
kotlin中 ?:的用法
android·开发语言·kotlin
charlie1145141917 小时前
嵌入式的现代C++教程——constexpr与设计技巧
开发语言·c++·笔记·单片机·学习·算法·嵌入式