(*(void(*) ())0)();------这是什么?

(*(void(*)())0)();

请问这条语句是什么?有什么作用?

当上一次面试官把这到题抛给我时,很遗憾,没能将这道题的完整答案讲给面试官。其实这个问题在《C陷阱和缺陷》和《C语言深度剖析》这两本书都有过阐述。因此,我重新去看了一遍,现在就将完整的过程写下来,一个是帮助我理解和记忆,另一个也是把答案共享给遇到相同问题的你们。下面就来进行分析:

第一步:void(*)(),可以明白这是一个函数指针类型。这个函数没有参数,没有返回值。

第二步:(void(*)())0,这是将0强制转换为函数指针类型,0是一个地址,也就是说一个函数存在首地址为0的一块区域内。

第三步:((void()())0),这是取0地址开始的一段内存里面的内容,其内容就是保存在首地址为0的一段区域内的函数。

第四步:((void()())0)(),这是函数调用。

最后加了一个分号,就变成了原来的样子,所以最终的结果是,这是一条函数调用语句。

这样的代码看起来不好理解,可以用typedef来帮助使表述更加清晰:

typedef void (*funcptr)();
(*(funcptr)0)();

如果觉得简单,这道题可以稍微再改改。

(*(char**(*)(char**,char**))0)(char**,char**);

其实要是明白了上一道题,估计大家都会分析了。下面我再写一下我的分析过程,写得多了自然就不会忘了。

第一步:char**(*)(char**,char**),首先可以看到这是一个函数指针类型。这个函数有两个参数,都是char**,返回值类型也是char**。

第二步:(char**(*)(char**,char**))0,这就是将0强转为指向函数的指针,0就成了一个地址值,也就是说这个函数存在首地址为0的一段内存内。函数名可以说就是一个地址,通过它我们可以调用执行函数体中的语句。所以0这个地址里面就存放了这个函数的首地址,0就是二级指针。

第三步:(*(char**(*)(char**,char**))0),这一步可以说是在解引用,也就是取0地址开始的一段内存里面的内容,其内容就是保存在首地址为0的一段区域内的函数,也就是函数的地址值,相当于函数名。

第四步:(*(char**(*)(char**,char**))0)(char**,char**)();,这是函数调用语句,意思是调用执行这个函数的代码块,而且函数的参数列表必须与强转时的函数参数列表一致,保证调用的是同一函数。

同样可以使用typedef来帮助我们理解代码:

typedef char** (*funcptr)(char**,char**);
(*(funcptr)0)(char**,char**);

当然要再举一个比较复杂的例子:

考虑signal库函数,这个函数在Linux进程间通信有用到,用来捕获和处理已经存在的信号,是一个整数。在包括该函数的C编译器实现中,signal函数接收两个参数:一个是代表需要“被捕获”的特定signal的整数值;另一个是指向用户提供的函数的指针,该函数用于处理“捕获到”的特定signal,返回值类型为void。

一般情况下,程序员并不主动声明signal函数,而是直接使用系统头文件signal.h中的声明。那么,在头文件signal.h中,signal函数是如何声明的呢?

首先,让我们从用户定义信号处理函数开始考虑,这无疑是最容易解决的。该函数可以定义如下:

void sigfunc(int n)
{
    //特定信号处理部分;
}

函数sigfunc的参数是一个代表特定信号的整数值,此处暂时忽略。

sigfunc函数的声明可以如下:

void sigunc(int);

假定我们希望声明一个指向sigfunc函数的指针变量,不妨命名为sfp。因为sfp指向sigfunc函数,则*sfp就代表了sigfunc函数,因此*sfp可以被调用。

又假定sig是一个整数,则(*sfp)(sig)的值为void类型,因此我们可以如下声明sfp:

void (*sfp)(int);

因为signal函数的返回值类型与sfp的返回值类型一样,上式就声明了signal函数,我们也可以如下声明signal函数

void (*signal(something))(int);

此处的something代表了signal函数的参数类型,我们还需要,我们还需要进一步了解如何声明它们。上面声明可以这样理解:传递适当地参数以调用signal函数,对signal函数返回值(为函数指针类型)解除引用,然后传递一个整型参数调用解除引用后所得参数,最后返回值为void类型。因此,signal函数的返回值是一个指向返回值为void类型的函数的指针。

那么,signal函数的参数又是如何呢?signal函数接受两个参数:一个整型的信号编号,以及一个指向用户定义的信号处理函数的指针。我们之前已经定义了指向用户定义的信号处理函数的指针sfp:

void (*sfp)(int);

sfp的类型可以通过将上面的声明中的sfp去掉而得到,即void (*)(int)。此外,signal函数的返回值是一个指向调用前的用户定义信号处理函数的指针,这个指针的类型与sfp的指针类型一致。因此,我们可以如下声明signal函数:

void (*signal(int,void(*)(int)))(int);

同样地,使用typdef可以简化上面的函数声明:

typedef void (*HANDLER)(int);
HANDER signal(int,HANDLER);
  • 12
    点赞
  • 32
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值