作者:夢凡
精通C語言??
本文展示了10個C語言的迷題以及答案,而且有相當(dāng)?shù)囊恍├涌赡苁俏覀內(nèi)粘9ぷ骺赡軙姷玫降摹Mㄟ^這些迷題,希望你能更了解C語言。
如果你不看答案,不知道是否有把握回答各個謎題?讓我們來試試。
會輸出"hello-std-out"?
#include
int main()
{
while (1)
{
fprintf(stdout, "hello-std-out");
fprintf(stderr, "hello-std-err");
sleep(1);
}
return 0;
}
參考答案:
stdout和stderr是不同設(shè)備描述符。stdout是塊設(shè)備,stderr則不是。對于塊設(shè)備,只有當(dāng)下面幾種情況下才會被輸入:遇到回車;緩沖區(qū)滿;flush被調(diào)用。而stderr則不會。
這段程序是有問題嗎?
?#include
int main()
{
int a = 1,2;
printf("a : %d\n",a);
return 0;
}
參考答案:
這個程序會得到編譯出錯(語法出錯)。逗號表達(dá)式是沒錯,可是在初始化和變量聲明時,逗號并不是逗號表達(dá)式的意義。這點(diǎn)要區(qū)分,要修改上面這個程序,你需要加上括號:"int a = (1,2);"。
下面的程序會有什么樣的輸出呢?
#include
int main()
{
int i=43;
printf("%d\n",printf("%d",printf("%d",i)));
return 0;
}
參考答案
程序會輸出4321,你知道為什么嗎?要知道為什么,你需要知道printf的返回值是什么。printf返回值是輸出的字符個數(shù)。
下面的程序會輸出什么??
#include
int main()
{
float a = 12.5;
printf("%d\n", a);
printf("%d\n", (int)a);
printf("%d\n", *(int*)&a);
return 0;
}
參考答案
該項程序輸出:"0 12 1095237632"。
原因是:浮點(diǎn)數(shù)是4個字節(jié),12.5f轉(zhuǎn)成二進(jìn)制是:01000001010010000000000000000000,十六進(jìn)制是:0x41480000,十進(jìn)制是:1095237632。所以,第二和第三個輸出相信大家也知道是為什么了。
而對于第一個,為什么會輸出0,我們需要了解一下float和double的內(nèi)存布局,如下:
-
float: 1位符號位(s)、8位指數(shù)(e),23位尾數(shù)(m,共32位)。
-
double: 1位符號位(s)、11位指數(shù)(e),52位尾數(shù)(m,共64位)。
然后,我們還需要了解一下printf由于類型不匹配,所以,會把float直接轉(zhuǎn)成double,注意,12.5的float和double的內(nèi)存二進(jìn)制完全不一樣。別忘了在x86芯片下使用是的反字節(jié)序,高位字節(jié)和低位字位要反過來。所以:
-
float版:0x41480000 (在內(nèi)存中是:00 00 48 41)。
-
double版:0x4029000000000000 (在內(nèi)存中是:00 00 00 00 00 00 29 40)。
而我們的%d要求是一個4字節(jié)的int,對于double的內(nèi)存布局,我們可以看到前四個字節(jié)是00,所以輸出自然是0了。這個示例向我們說明printf并不是類型安全的,這就是為什么C++要引如cout的原因了。
下面的程序輸出是多少?并解釋為什么?
?#include
int 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;
}
參考答案
該程序在編譯時,報錯:“b”的初始化操作由“case”標(biāo)簽跳過,“default”標(biāo)簽跳過“b”的初始化操作
下面的程序會有什么潛在的危險?
?#include
int main()
{
char str[10];
printf("Enter the string:");
scanf("%s", str);
printf("You entered:%s\n", str);
return 0;
}
參考答案
本題很簡單了。這個程序的潛在問題是,如果用戶輸入了超過80個長度的字符,那么就會有數(shù)組越界的問題了,你的程序很有可能會crash了。
請問下面的程序輸出什么?
?#include
int 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;
}
參考答案
如果你覺得輸出分別是:10,4,11。那么你就錯了。
錯在了第三個,第一個是10沒有什么問題,第二個是4,也沒有什么問題,因為是32位機(jī)上一個int有4個字節(jié)。但是第三個為什么輸出的不是11呢?居然還是10?原因是,sizeof不是一個函數(shù),是一個操作符,其求i++的類型的size,這是一件可以在程序運(yùn)行前(編譯時)完全的事情,所以,sizeof(i++)直接就被4給取代了,在運(yùn)行時也就不會有了i++這個表達(dá)式。
下面的程序的輸出值是什么?
?#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;
}
參考答案
如果你對于PrintInt這個宏有問題的話,可以去看一看資料。不過,本例的問題不在這里,本例的輸出會是:1,8,64,1000。其實很簡單了,在C/C++中,以0開頭的數(shù)字都是八進(jìn)制的。
下面的輸出是什么?
?#include
int main()
{
int i = 6;
if (((++i < 7) && (i++ / 6)) || (++i <= 9));
printf("%d\n", i);
return 0;
}
參考答案
本題并不簡單的是考前綴++或反綴++,本題主要考的是&&和||的短路求值的問題。
所謂短路求值:對于(條件1 && 條件2),如果“條件1”是false,那“條件2”的表達(dá)式會被忽略了。對于(條件1 || 條件2),如果“條件1”為true,而“條件2”的表達(dá)式則被忽略了。
所以,我相信你會知道本題的答案是什么了。
下面的C程序是合法的嗎?
?#include
int main()
{
int a = 3, b = 5;
printf(&a["Ya!Hello! how is this? %s\n"], &b["junk/super"]);
//等價printf("Hello! how is this? %s\n", "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] 其實就是 *(a+i)也就是 *(i+a),所以如果寫成 i[a] 應(yīng)該也不難理解了。
免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺僅提供信息存儲服務(wù)。文章僅代表作者個人觀點(diǎn),不代表本平臺立場,如有問題,請聯(lián)系我們,謝謝!