我被這道c語言筆試題整不會(huì)了
半個(gè)月前這位同學(xué)找到我,他當(dāng)時(shí)準(zhǔn)備參加面試,就和我聊了很多,總體感覺這位同學(xué)基礎(chǔ)還是不錯(cuò)。我那時(shí)候也就順便給他找了一些C語言、Linux等等筆試題,后面也教他稍微修改過幾次簡(jiǎn)歷??傮w來說,這位同學(xué)能有現(xiàn)在的三個(gè)比較好的offer很大原因在于他自身的基礎(chǔ)不錯(cuò),肯付出、肯努力的結(jié)果。前幾天和我分享了他參加面試遇到的一道c語言筆試題,他說當(dāng)時(shí)有點(diǎn)整不會(huì)了,竟然做錯(cuò)了,然后面試官也讓他自己回去想想。因?yàn)榇蟛糠诸}做的還可以,最后也通過了面試。題目是這樣的:
一般而言,我們都會(huì)認(rèn)為任何指針p,對(duì)其*操作的*p的意思就是對(duì)p解引用,也就是取p的目標(biāo),對(duì)于指針的知識(shí)在文章(看完還不會(huì)指針,錘自己!)有講到過,一起看看上面的題目輸出的結(jié)果:
當(dāng)然編譯會(huì)有警告,因?yàn)轭愋筒灰粯樱瑥?qiáng)制轉(zhuǎn)換一下就行,或者直接先不理。
我們分析下這三個(gè)結(jié)果:
1. a的值是0x100,這個(gè)沒毛病。
2. p的值是0xe532deec,而0xe532deec是變量a的地址,因?yàn)橹羔榩存放的就是a的地址,也沒什么毛病。可以加一行(printf("&a:%#x\n", &a); )代碼,看看a的地址是否和輸出的p是不是一樣的。
3. *p的輸出結(jié)果有點(diǎn)不太理解,指針p存放的是a的地址,對(duì)指針進(jìn)行*操作,*p就是取出a的值,那么*p輸出的值應(yīng)該是0x100才對(duì),但結(jié)果怎么還是a的地址呢?對(duì)指針p進(jìn)行*操作怎么不起作用?
是不是很怪,p和*p竟然是一樣的!
有沒有一種感覺:
自以為對(duì)指針的基本操作已經(jīng)掌握,但是看到這道面試題,是不是被狠狠扇了一下。
說實(shí)話這完全是坑爹不實(shí)際的筆試題,不過這樣坑爹的筆試題還有很多!
造成這個(gè)問題的原因其實(shí)很簡(jiǎn)單,問題出在這行代碼:void (*p)(void);
仔細(xì)看,這里的指針p其實(shí)是一個(gè)函數(shù)指針,函數(shù)指針也是指針。
所謂的函數(shù),在內(nèi)存中是一段可執(zhí)行的代碼,函數(shù)指針就是指向函數(shù)的入口地址(首地址)。但是在c/c++中,函數(shù)指針解引用還是會(huì)當(dāng)成函數(shù)指針處理,所以函數(shù)指針可以不限次數(shù)的解引用,效果和不解引用是一樣的,可以換句話說:*操作對(duì)它是無效的!
我們把代碼改一改再看:
void func(int a){ printf("a = %d\n", a);} int main(){ void (*p)(int a); p = &func; // p = func; 這樣寫也是可以的,函數(shù)名其實(shí)就相當(dāng)于這個(gè)函數(shù)的入口地址(首地址) p(10); // 相當(dāng)于調(diào)用了函數(shù)func(10) (*p)(10); // 相當(dāng)于調(diào)用了函數(shù)func(10)}
我們使用指針p來調(diào)用函數(shù)func,可以發(fā)現(xiàn)既可以使用*操作來解引用,也可以什么都不寫,效果會(huì)是一樣的,我們看看輸出結(jié)果:
最后看一個(gè)非常直觀有趣的代碼:
int main(){ int a = 1; printf("a: %d\n", a); (*printf)("a: %d\n", a); (**printf)("a: %d\n", a);}
常用的printf我們已經(jīng)深入骨髓!融入血液!但是(*printf)和(**printf)你未必見過,和前面一樣,*操作對(duì)printf函數(shù)無效,它們都可以編譯,結(jié)果都是一樣的,可以動(dòng)手試一試。
ok,就分享到這里,有沒有感覺學(xué)海無涯。