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