我被这道c语言笔试题整不会了
扫描二维码
随时随地手机看文章
半个月前这位同学找到我,他当时准备参加面试,就和我聊了很多,总体感觉这位同学基础还是不错。我那时候也就顺便给他找了一些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,就分享到这里,有没有感觉学海无涯。