一、信号函数的理解

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))

最后修改:2018 年 05 月 21 日
如果觉得我的文章对你有用,请随意赞赏