一、信号函数的理解
C语言中信号函数的原型为:
void (*signal(int signo, void (*func)(int)))(int);
这个函数定义看起来十分复杂,可以分为以下两步来理解:
首先看signal(int signo, void (*func)(int))
部分,signal
是一个函数,它的形参为一个int类型的signo和一个函数指针func。
除去形参部分后剩下的就是返回值了,即void (*)(int)
这部分是返回值,它是一个函数指针——这个函数形参类型为int,没有返回值。
所以总体来看,signal
是一个形参为“int类型和函数指针(形参为int类型,返回空)”,返回值为“函数指针(形参为int,返回空)”的函数。
为了简化该函数,可以有以下定义:
typedef void sig_func(int); // sig_func是一个函数,它的形参为int类型,返回空。
在这个基础上,上面的函数可以改写成:
sig_func *signal(int signo, sig_func *func);
二、信号常量的理解
上面的函数在返回错误时将返回SIG_ERR,SIG_ERR和SIG_ERR这几个常量,它们的定义为:
#define SIG_ERR (void (*)())-1
#define SIG_DFL (void (*)())0
#define SIG_IGN (void (*)())1
在上面可以看到信号函数的返回值是一个函数指针,其类型为void (*)(int)
,但是这里的常量却是void (*)()
类型,和返回值类型不同,WHY?
原因:c语言中允许以下形式的函数声明和定义。
int f(); // 声明
int f(int a, int b) { // 定义
// ...
}
例如以下代码:
#include <stdio.h>
int f();
int main() {
printf("%d\n", f(1, 2));
return 0;
}
int f(int x, int y) { // 函数
return x + y;
}
程序能正常编译运行,输出结果3
。不过如果把main里面的f(1, 2)
改成f()
程序也能运行,只是结果是随机值。
所以对于#define SIG_ERR (void (*)())-1
来说,(void (*)())
可以看作返回值类型的声明,对-1
取地址然后返回,兼容于(void (*)(int))
。
此处评论已关闭