C语言太简单?这14道C语言谜题,你能答对几个?
本文展示了14個(gè)C語(yǔ)言的迷題以及答案,代碼應(yīng)該是足夠清楚的,而且有相當(dāng)?shù)囊恍├涌赡苁俏覀內(nèi)粘9ぷ骺赡軙?huì)見得到的。通過(guò)這些迷題,希望你能更了解C語(yǔ)言。
如果你不看答案,不知道是否有把握回答各個(gè)謎題?讓我們來(lái)試試。
1
下面的程序并不見得會(huì)輸出"hello-std-out",你知道為什么嗎?
#include#includeint main(){ while(1) { fprintf(stdout,"hello-std-out"); fprintf(stderr,"hello-std-err"); sleep(1); } return 0;}參考答案
stdout和stderr是不同設(shè)備描述符。stdout是塊設(shè)備,stderr則不是。對(duì)于塊設(shè)備,只有當(dāng)下面幾種情況下才會(huì)被輸入:遇到回車;緩沖區(qū)滿;flush被調(diào)用。而stderr則不會(huì)。
2
下面的程序看起來(lái)是正常的,使用了一個(gè)逗號(hào)表達(dá)式來(lái)做初始化。可惜這段程序是有問(wèn)題的。你知道為什么嗎?
#includeint main(){ int a = 1,2; printf("a : %d\n",a); return 0;}參考答案
這個(gè)程序會(huì)得到編譯出錯(cuò)(語(yǔ)法出錯(cuò))。逗號(hào)表達(dá)式是沒(méi)錯(cuò),可是在初始化和變量聲明時(shí),逗號(hào)并不是逗號(hào)表達(dá)式的意義。這點(diǎn)要區(qū)分,要修改上面這個(gè)程序,你需要加上括號(hào):"int a = (1,2);"。
3
下面的程序會(huì)有什么樣的輸出呢?
#includeint main(){ int i=43; printf("%d\n",printf("%d",printf("%d",i))); return 0;}參考答案
程序會(huì)輸出4321,你知道為什么嗎?要知道為什么,你需要知道printf的返回值是什么。printf返回值是輸出的字符個(gè)數(shù)。
4
下面的程序會(huì)輸出什么?
#includeint main(){ float a = 12.5; printf("%d\n", a); printf("%d\n", (int)a); printf("%d\n", *(int *)&a); return 0;}參考答案
該項(xiàng)程序輸出:"0 12 1095237632"。
原因是:浮點(diǎn)數(shù)是4個(gè)字節(jié),12.5f轉(zhuǎn)成二進(jìn)制是:01000001010010000000000000000000,十六進(jìn)制是:0x41480000,十進(jìn)制是:1095237632。所以,第二和第三個(gè)輸出相信大家也知道是為什么了。
而對(duì)于第一個(gè),為什么會(huì)輸出0,我們需要了解一下float和double的內(nèi)存布局,如下:
? float: 1位符號(hào)位(s)、8位指數(shù)(e),23位尾數(shù)(m,共32位)。
? double: 1位符號(hào)位(s)、11位指數(shù)(e),52位尾數(shù)(m,共64位)。
然后,我們還需要了解一下printf由于類型不匹配,所以,會(huì)把float直接轉(zhuǎn)成double,注意,12.5的float和double的內(nèi)存二進(jìn)制完全不一樣。別忘了在x86芯片下使用是的反字節(jié)序,高位字節(jié)和低位字位要反過(guò)來(lái)。所以:
? float版:0x41480000 (在內(nèi)存中是:00 00 48 41)。
? double版:0x4029000000000000 (在內(nèi)存中是:00 00 00 00 00 00 29 40)。
而我們的%d要求是一個(gè)4字節(jié)的int,對(duì)于double的內(nèi)存布局,我們可以看到前四個(gè)字節(jié)是00,所以輸出自然是0了。這個(gè)示例向我們說(shuō)明printf并不是類型安全的,這就是為什么C++要引如cout的原因了。
5
下面,我們?cè)賮?lái)看一個(gè)交叉編譯的事情,下面的兩個(gè)文件可以編譯通過(guò)嗎?如果可以通過(guò),結(jié)果是什么?
//file1.cint arr[80];//file2.cextern int *arr;int main(){ arr[1] = 100; printf("%d\n", arr[1]); return 0;}參考答案
該程序可以編譯通過(guò),但運(yùn)行時(shí)會(huì)出錯(cuò)。為什么呢?原因是,在另一個(gè)文件中用 extern int *arr來(lái)外部聲明一個(gè)數(shù)組并不能得到實(shí)際的期望值,因?yàn)樗麄兊念愋筒⒉黄ヅ洹K詫?dǎo)致指針實(shí)際并沒(méi)有指向那個(gè)數(shù)組。
注意:一個(gè)指向數(shù)組的指針,并不等于一個(gè)數(shù)組。
修改:"extern int arr[]"。
6
請(qǐng)說(shuō)出下面的程序輸出是多少?并解釋為什么?(注意,該程序并不會(huì)輸出"b is 20")
#includeint main(){ int a=1; switch(a) { int b=20; case 1: printf("b is %d\n",b); break; default: printf("b is %d\n",b); break; } return 0;}參考答案
該程序在編譯時(shí),可能會(huì)出現(xiàn)一條warning: unreachable code at beginning of switch statement。我們以為進(jìn)入switch后,變量b會(huì)被初始化,其實(shí)并不然,因?yàn)閟witch-case語(yǔ)句會(huì)把變量b的初始化直接就跳過(guò)了。所以,程序會(huì)輸出一個(gè)隨機(jī)的內(nèi)存值。
7
請(qǐng)問(wèn)下面的程序會(huì)有什么潛在的危險(xiǎn)?
#includeint main(){ char str[80]; printf("Enter the string:"); scanf("%s",str); printf("You entered:%s\n",str); return 0;}參考答案
本題很簡(jiǎn)單了。這個(gè)程序的潛在問(wèn)題是,如果用戶輸入了超過(guò)80個(gè)長(zhǎng)度的字符,那么就會(huì)有數(shù)組越界的問(wèn)題了,你的程序很有可能會(huì)crash了。
8
請(qǐng)問(wèn)下面的程序輸出什么?
#includeint main(){ int i; i = 10; printf("i : %d\n",i); printf("sizeof(i++) is: %d\n",sizeof(i++)); printf("i : %d\n",i); return 0;}參考答案
如果你覺(jué)得輸出分別是:10,4,11。那么你就錯(cuò)了。
錯(cuò)在了第三個(gè),第一個(gè)是10沒(méi)有什么問(wèn)題,第二個(gè)是4,也沒(méi)有什么問(wèn)題,因?yàn)槭?2位機(jī)上一個(gè)int有4個(gè)字節(jié)。但是第三個(gè)為什么輸出的不是11呢?居然還是10?原因是,sizeof不是一個(gè)函數(shù),是一個(gè)操作符,其求i++的類型的size,這是一件可以在程序運(yùn)行前(編譯時(shí))完全的事情,所以,sizeof(i++)直接就被4給取代了,在運(yùn)行時(shí)也就不會(huì)有了i++這個(gè)表達(dá)式。
9
請(qǐng)問(wèn)下面的程序的輸出值是什么?
#include#include #define SIZEOF(arr) (sizeof(arr)/sizeof(arr[0]))#define?PrintInt(expr)?printf("%s:%d\n",#expr,(expr)) int main(){/* The powers of 10 */ int pot[] = { 0001, 0010, 0100, 1000??????};int i; for(i=0;i<SIZEOF(pot);i++)PrintInt(pot[i]); return 0;}參考答案
如果你對(duì)于PrintInt這個(gè)宏有問(wèn)題的話,可以去看一看資料。不過(guò),本例的問(wèn)題不在這里,本例的輸出會(huì)是:1,8,64,1000。其實(shí)很簡(jiǎn)單了,在C/C++中,以0開頭的數(shù)字都是八進(jìn)制的。
10
請(qǐng)問(wèn)下面的程序輸出是什么?(絕對(duì)不是10)
#include#define PrintInt(expr) printf("%s : %dn",#expr,(expr))int main(){ int y = 100; int *p; p = malloc(sizeof(int)); *p = 10; y = y/*p; /*dividing y by *p */; PrintInt(y); return 0;}參考答案
本題輸出的是100。為什么呢?問(wèn)題就出在"y = y/*p;"上了,我們本來(lái)想的是"y / (*p)",然而,我們沒(méi)有加入空格和括號(hào),結(jié)果"y/*p"中的"/*"被解釋成了注釋的開始。于是,這也是整個(gè)惡夢(mèng)的開始。
11
下面的輸出是什么?
#includeint main(){ int i = 6; if( ((++i < 7) && ( i++/6)) || (++i <= 9)); printf("%d\n",i); return 0;}參考答案
本題并不簡(jiǎn)單的是考前綴++或反綴++,本題主要考的是&&和||的短路求值的問(wèn)題。
所謂短路求值:對(duì)于(條件1 && 條件2),如果“條件1”是false,那“條件2”的表達(dá)式會(huì)被忽略了。對(duì)于(條件1 || 條件2),如果“條件1”為true,而“條件2”的表達(dá)式則被忽略了。
所以,我相信你會(huì)知道本題的答案是什么了。
12
下面的C程序是合法的嗎?如果是,那么輸出是什么?
#includeint main(){ int a=3, b = 5; printf(&a["Ya!Hello! how is this? %s\n"], &b["junk/super"]); printf(&a["WHAT%c%c%c %c%c %c !\n"], 1["this"],2["beauty"],0["tool"],0["is"],3["sensitive"],4["CCCCCC"]); return 0;}參考答案
本例是合法的,輸出為:"Hello! how is this? super That is C !"
本例主要展示了一種另類的用法。下面的兩種用法是相同的:
? "hello"[2]
? 2["hello"]
如果你知道:a[i] 其實(shí)就是 *(a+i)也就是 *(i+a),所以如果寫成 i[a] 應(yīng)該也不難理解了。
13
請(qǐng)問(wèn)下面的程序輸出什么?(假設(shè):輸入"Hello, World")
#includeint main(){ char dummy[80]; printf("Enter a string:\n"); scanf("%[^r]",dummy); printf("%s\n",dummy); return 0;}參考答案
本例的輸出是"Hello, Wo"。scanf中的"%[^r]"是從中作梗的東西,意思是遇到字符r就結(jié)束了。
14
下面的程序試圖使用"位操作"來(lái)完成"乘5"的操作,不過(guò)這個(gè)程序中有個(gè)BUG,你知道是什么嗎?
#include#define PrintInt(expr) printf("%s : %d\n",#expr,(expr))int FiveTimes(int a){ int t; t = a<<2 + a; return t;}int main(){ int a = 1, b = 2,c = 3; PrintInt(FiveTimes(a)); PrintInt(FiveTimes(b)); PrintInt(FiveTimes(c)); return 0;}參考答案
本題的問(wèn)題在于函數(shù)FiveTimes中的表達(dá)式"t = a<<2 + a;"。對(duì)于a<<2這個(gè)位操作,優(yōu)先級(jí)要比加法要低,所以這個(gè)表達(dá)式就成了"t = a << (2+a)",于是我們就得不到我們想要的值。
該程序修正如下:
int FiveTimes(int a){ ? ? ?int t; ? ? ?t = (a<<2) + a; ? ? ?return t;}*本文系網(wǎng)絡(luò)轉(zhuǎn)載,版權(quán)歸原作者所有,如有侵權(quán)請(qǐng)聯(lián)系刪除
-END-
最 后??
?若覺(jué)得文章對(duì)你有幫助,隨手轉(zhuǎn)發(fā)分享,也是我們繼續(xù)更新的動(dòng)力。
送100G編程資源,涵蓋C/C++、Python、Linux、JAVA、數(shù)據(jù)結(jié)構(gòu)、算法等入門基礎(chǔ)資料
長(zhǎng)按二維碼識(shí)別,安安同學(xué)和大家一起開始學(xué)編程
創(chuàng)作挑戰(zhàn)賽新人創(chuàng)作獎(jiǎng)勵(lì)來(lái)咯,堅(jiān)持創(chuàng)作打卡瓜分現(xiàn)金大獎(jiǎng)總結(jié)
以上是生活随笔為你收集整理的C语言太简单?这14道C语言谜题,你能答对几个?的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 长城汽车:四驱电混技术Hi4,系统功率可
- 下一篇: C语言三剑客之《C陷阱与缺陷》一书精华提