当前位置:首页 » 《随便一记》 » 正文

【C陷阱与缺陷】两道“有趣”的代码题

25 人参与  2023年04月29日 13:26  分类 : 《随便一记》  评论

点击全文阅读


在这里插入图片描述

在这里插入图片描述

如果读者不了解函数指针的,可以先看看这篇文章 链接

< 第一题 >

代码:

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

解析:

?如果你是头一次看上面这段代码的话,心里一定是一个大大的问号???现在我就来解释一下

本题的突破口在于这个0,仔细观察可以发现,0前面有一个括号(),括号里面的这种形式若是你自己去看的话就是一个函数指针,那相当于就是对0进行一个强制类型转换,把它变成一个函数地址,然后前面的*我们刚才讲过,就是对这个函数进行解引用,获取到这个函数。那么最后一步便是去调用这个函数

具体的分解可以看看下图?

在这里插入图片描述
分步细说:

void (*)() —— 》一个没有形参,返回类型为void的函数指针
(void (*)())0 ——》 对0进行强制类型转换,使其被解释成为一个函数的地址
*(void (*)())0 ——》对0地址处的函数进行解引用,获取到这个函数
(*(void (*)())0)() ——》调用0地址处的函数

原文现身:
在这里插入图片描述

< 第二题 >

代码:

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

解析:

?同理,若是第一次见一定会被它绕晕了?了

本题真的可以说是在套娃了,首先你看到的一定是signal,它呢是C语言中的一个信号函数,有兴趣可以去了解一下,我们知道()的优先级高于*,所以signal会和后面的内容先结合,那其实已经可以看出这是一个函数声明了。进到里面再来看看这个函数有两个参数,一个是int,一个是函数指针,那么外层的又是什么呢? 仔细看下图,我将内部的signal()函数声明抽离了出来,只剩下了头和尾,你可以做一个视觉上的合并,那其实又是一个void (*)(int)的函数指针,其实这就是signal函数的返回类型,是一个函数指针

在这里插入图片描述

同样地,我们再来捋一遍

分步细说:

void (*)(int) —— 》是一个函数指针,为signal函数的形参
signal(int, void(*)(int)) ——》 是一个函数声明,signal与右侧的()率先结合,内部有两个形参
void (*)(int) ——》也是一个函数指针,不过是作为signal函数返回类型

优化:

对于上面的这种写法你是否觉得很冗余,其实可以再度进行一个优化,那么你可能很快就看得懂了

因为 void (*)(int) 是出现了两次,之前我们在C语言中有学习过typedef这个关键字,可以用来对一个很长的数据类型或者变量进行重命名,那么在这里我们也可以这样做不过呢,你要把重命名后的名字放在(*)里面,因为语法这么规定了,去掉变量名后就是它的类型
typedef void(*ptr_t)(int);
于是这句代码就可以简化为下面这种形式?注意解引用那个*不要了,函数指针这里是可以省略的
//void (*signal(int, void(*)(int)))(int);ptr_t signal(int, ptr_t);

原文现身:

在这里插入图片描述


两道题讲完了,你是否有收获呢?有兴趣的话可以私信我,发你《C陷阱与缺陷》的电子书.pdf

—— 2023.4.18记

在这里插入图片描述


点击全文阅读


本文链接:http://zhangshiyu.com/post/60130.html

<< 上一篇 下一篇 >>

  • 评论(0)
  • 赞助本站

◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。

关于我们 | 我要投稿 | 免责申明

Copyright © 2020-2022 ZhangShiYu.com Rights Reserved.豫ICP备2022013469号-1