UNIX网络编程笔记 信号处理
在 C 语言的标准库中并没有直接定义 Sigfunc。
Sigfunc 实际上是《Unix 环境高级编程》(APUE)这本经典著作的作者 W. Richard Stevens 为了简化信号处理代码而自定义的一个类型定义(typedef)。
由于信号处理函数(Signal Handler)的原型比较复杂,使用 Sigfunc 可以大大提高代码的可读性。
- 为什么需要 Sigfunc?在标准 C 库中,signal 函数的原型非常"反人类",读起来很吃力:
c
void (*signal(int \text{ signo}, void (*\text{func})(int)))(int);
这行代码的意思是:signal 是一个函数,它接收一个整型 signo 和一个指向"接收 int 返回 void"的函数指针 func,并返回一个指向"接收 int 返回 void"的函数指针。
- Sigfunc 是如何定义的?
为了简化,Stevens 在他的头文件(通常是 apue.h)中这样定义:
c
typedef void (*Sigfunc)(int);
- 使用 Sigfunc 后的效果
一旦有了这个定义,signal 函数的原型就可以写得非常简洁清晰:
c
Sigfunc *signal(int signo, Sigfunc *func);
这段代码的逻辑一目了然:signal 函数接收一个信号编号和一个处理函数,返回旧的处理函数。
- 如何在自己的程序中使用?
如果你想在自己的项目中使用这种风格,你需要手动添加这个定义。以下是一个完整的用法示例:
c
#include <stdio.h>
#include <unistd.h>
#include <signal.h>
// 1. 手动定义 Sigfunc 类型
typedef void (*Sigfunc)(int);
// 2. 实现一个符合 Sigfunc 原型的处理函数
void my_handler(int signo) {
if (signo == SIGINT) {
printf("\n接收到 Ctrl+C (SIGINT),程序即将安全退出...\n");
}
}
int main() {
// 3. 使用 Sigfunc 定义的类型来调用 signal
Sigfunc old_handler = signal(SIGINT, my_handler);
printf("程序正在运行,请按 Ctrl+C 测试信号...\n");
while(1) {
pause(); // 等待信号
}
return 0;
}