查看原文
其他

我被这道c语言笔试题整不会了

Linux_Daily 混说Linux 2022-11-19

前几天收到一位同学的好消息,收到了三家公司的offer。



半个月前这位同学找到我,他当时准备参加面试,就和我聊了很多,总体感觉这位同学基础还是不错。我那时候也就顺便给他找了一些C语言、Linux等等笔试题,后面也教他稍微修改过几次简历。


总体来说,这位同学能有现在的三个比较好的offer很大原因在于他自身的基础不错,肯付出、肯努力的结果。


前几天和我分享了他参加面试遇到的一道c语言笔试题,他说当时有点整不会了,竟然做错了,然后面试官也让他自己回去想想。因为大部分题做的还可以,最后也通过了面试。


题目是这样的:



一般而言,我们都会认为任何指针p,对其*操作的*p的意思就是对p解引用,也就是取p的目标,对于指针的知识在文章(看完还不会指针,锤自己!)有讲到过,一起看看上面的题目输出的结果:



当然编译会有警告,因为类型不一样,强制转换一下就行,或者直接先不理。





我们分析下这三个结果:

  1. a的值是0x100,这个没毛病。


  2. p的值是0xe532deec,而0xe532deec是变量a的地址,因为指针p存放的就是a的地址,也没什么毛病。可以加一行(printf("&a:%#x\n", &a); )代码,看看a的地址是否和输出的p是不是一样的。


  3. *p的输出结果有点不太理解,指针p存放的是a的地址,对指针进行*操作,*p就是取出a的值,那么*p输出的值应该是0x100才对,但结果怎么还是a的地址呢?对指针p进行*操作怎么不起作用?




是不是很怪,p和*p竟然是一样的!

有没有一种感觉:

       自以为对指针的基本操作已经掌握,但是看到这道面试题,是不是被狠狠扇了一下。

说实话这完全是坑爹不实际的笔试题,不过这样坑爹的笔试题还有很多!


造成这个问题的原因其实很简单,问题出在这行代码:void (*p)(void); 


仔细看,这里的指针p其实是一个函数指针函数指针也是指针


所谓的函数,在内存中是一段可执行的代码,函数指针就是指向函数的入口地址(首地址)。但是在c/c++中,函数指针解引用还是会当成函数指针处理,所以函数指针可以不限次数的解引用,效果和不解引用是一样的,可以换句话说:*操作对它是无效的!


我们把代码改一改再看:

void func(int a){ printf("a = %d\n", a);}
int main(){ void (*p)(int a);  p = &func; // p = func;  这样写也是可以的,函数名其实就相当于这个函数的入口地址(首地址) p(10); // 相当于调用了函数func(10) (*p)(10); // 相当于调用了函数func(10)}


我们使用指针p来调用函数func,可以发现既可以使用*操作来解引用,也可以什么都不写,效果会是一样的,我们看看输出结果:



最后看一个非常直观有趣的代码:

int main(){ int a = 1; printf("a: %d\n", a); (*printf)("a: %d\n", a); (**printf)("a: %d\n", a);}


常用的printf我们已经深入骨髓!融入血液!但是(*printf)和(**printf)你未必见过,和前面一样,*操作对printf函数无效,它们都可以编译,结果都是一样的,可以动手试一试。


ok,就分享到这里,有没有感觉学海无涯。







关注微信公众号『混说Linux』,后台点击 关于混说 即可添加作者微信。


往期推荐

看完还不会指针,锤自己!

电路中的GND,它的本质是什么?

赶紧收藏!u-boot代码分析与移植

Linux为什么使用内核模块?


您可能也对以下帖子感兴趣

文章有问题?点此查看未经处理的缓存