C語言迷題

2021-06-21 15:06:36 字數 4278 閱讀 3023

摘錄至陳浩前輩部落格的c語言迷題,以備學習。

1、下面的程式並不見得會輸出 hello-std-out,你知道為什麼嗎?

#include #include int main()  

return 0;

}

參***:stdout和stderr是不是同裝置描述符。stdout是塊裝置,stderr則不是。對於塊裝置,只有當下面幾種情況下才會被輸入,1)遇到回車,2)緩衝區滿,3)flush被呼叫。而stderr則不會。

2、下面的程式看起來是正常的,使用了乙個逗號表示式來做初始化。可惜這段程式是有問題的。你知道為什麼呢?

#include int main()

參***:這個程式會得到編譯出錯(語法出錯),逗號表示式是沒錯,可是在初始化和變數宣告時,逗號並不是逗號表示式的意義。這點要區分,要修改上面這個程式,你需要加上括號: int a = (1,2);

3、下面的程式會有什麼樣的輸出呢?

#include int main()

參***:程式會輸出4321,你知道為什麼嗎?要知道為什麼,你需要知道printf的返回值是什麼。printf返回值是輸出的字元個數。

4、下面的程式會輸出什麼?

#include int main()  

參***:該項程式輸出如下所示, 0 12 1095237632 原因是:浮點數是4個位元組,12.5f 轉成二進位制是:01000001010010000000000000000000,十六進製制是:0x41480000,十進位制是:1095237632。所以,第二和第三個輸出相信大家也知道是為什麼了。而對於第乙個,為什麼會輸出0,我們需要了解一下float和double的記憶體布局,如下:

然後,我們還需要了解一下printf由於型別不匹配,所以,會把float直接轉成double,注意,12.5的float和double的記憶體二進位制完全不一樣。別忘了在x86晶元下使用是的反位元組序,高位位元組和低位字位要反過來。所以:

而我們的%d要求是乙個4位元組的int,對於double的記憶體布局,我們可以看到前四個位元組是00,所以輸出自然是0了。 這個示例向我們說明printf並不是型別安全的,這就是為什麼c++要引如cout的原因了。

5、下面,我們再來看乙個交叉編譯的事情,下面的兩個檔案可以編譯通過嗎?如果可以通過,結果是什麼?

file1.c

int arr[80];
file2.c

extern int *arr;

int main()

參***:該程式可以編譯通過,但執行時會出錯。為什麼呢?原因是,在另乙個檔案中用 extern int *arr來外部宣告乙個陣列並不能得到實際的期望值,因為他們的型別並不匹配。所以導致指標實際並沒有指向那個陣列。注意:乙個指向陣列的指標,並不等於乙個陣列。修改:extern int arr。(參考:iso c語言 6.5.4.2 節)

6、請說出下面的程式輸出是多少?並解釋為什麼?(注意,該程式並不會輸出 "b is 20")

#include int main()  

return 0;

}

參***:該程式在編譯時,可能會出現一條warning: unreachable code at beginning of switch statement。我們以為進入switch後,變數b會被初始化,其實並不然,因為switch-case語句會把變數b的初始化直接就跳過了。所以,程式會輸出乙個隨機的記憶體值。

7、請問下面的程式會有什麼潛在的危險?

#include int main()  

參***:本題很簡單了。這個程式的潛在問題是,如果使用者輸入了超過80個長度的字元,那麼就會有陣列越界的問題了,你的程式很有可以及會crash了。

8、請問下面的程式輸出什麼?

#include int main()  

參***:如果你覺得輸出分別是,10,4,11,那麼你就錯了,錯在了第三個,第乙個是10沒有什麼問題,第二個是4,也沒有什麼問題,因為是32位機上乙個int有4個位元組。但是第三個為什麼輸出的不是11呢?居然還是10?原因是,sizeof不是乙個函式,是乙個操作符,其求i++的型別的size,這是一件可以在程式執行前(編譯時)完全的事情,所以,sizeof(i++)直接就被4給取代了,在執行時也就不會有了i++這個表示式。

9、請問下面的程式的輸出值是什麼?

#include #include #define sizeof(arr) (sizeof(arr)/sizeof(arr[0]))

#define printint(expr) printf("%s:%d/n",#expr,(expr))

int main()

; int i;

for(i=0;i

參***:好吧,如果你對於printint這個巨集有問題的話,你可以去看一看《語言的歧義》中的第四個示例。不過,本例的問題不在這裡,本例的輸出會是:1,8,64,1000,其實很簡單了,以c/c++中,以0開頭的數字都是八進位制的。

10、請問下面的程式輸出是什麼?(絕對不是10)

#include #define printint(expr) printf("%s : %dn",#expr,(expr))

int main()

參***:本題輸出的是100。為什麼呢?問題就出在 y = y/*p;上了,我們本來想的是 y / (*p) ,然而,我們沒有加入空格和括號,結果y/*p中的 /*被解釋成了注釋的開始。於是,這也是整個惡夢的開始。

11、下面的輸出是什麼?

#include int main()  

參***:本題並不簡單的是考字首++或反綴++,本題主要考的是&&和||的短路求值的問題。所為短路求值:對於(條件1 && 條件2),如果「條件1」是false,那「條件2」的表示式會被忽略了。對於(條件1 || 條件2),如果「條件1」為true,而「條件2」的表示式則被忽略了。所以,我相信你會知道本題的答案是什麼了。

12、下面的c程式是合法的嗎?如果是,那麼輸出是什麼?

#include int main()  

參***:本例是合法的,輸出如下:

hello! how is this? super that is c !
本例主要展示了一種另類的用法。下面的兩種用法是相同的:

"hello"[2] 2["hello"]
如果你知道:a[i] 其實就是 *(a+i)也就是 *(i+a),所以如果寫成 i[a] 應該也不難理解了。

13、請問下面的程式輸出什麼?(假設:輸入 hello, world)

#include int main()  

參***:本例的輸出是「hello, wo」,scanf中的"%[^r]"是從中作梗的東西。意思是遇到字元r就結束了。

14、下面的程式試圖使用「位操作」來完成「乘5」的操作,不過這個程式中有個bug,你知道是什麼嗎?

#include #define printint(expr) printf("%s : %d/n",#expr,(expr))

int fivetimes(int a)

int main()

參***:本題的問題在於函式fivetimes中的表示式「t = a<<2 + a;」,對於a<<2這個位操作,優先順序要比加法要低,所以這個表示式就成了「t = a << (2+a)」,於是我們就得不到我們想要的值。該程式修正如下:

int fivetimes(int a)  

C 輕鬆解決世紀迷題

下面的問題相信很多人都聽過 1 有五棟五種顏色的房子 2 每一位房子的主人國籍都不同 3 這五個人每人只喝一種飲料,只抽一種牌子的香菸,只養一種寵物 4 沒有人有相同的寵物,抽相同牌子的香菸,喝相同的飲料 提示 1 英國人住在紅房子裡 2 瑞典人養了一條狗 3 丹麥人喝茶 4 綠房子在白房子左邊 5...

C 輕鬆解決世紀迷題

下面的問題相信很多人都聽過 1 有五棟五種顏色的房子 2 每一位房子的主人國籍都不同 3 這五個人每人只喝一種飲料,只抽一種牌子的香菸,只養一種寵物 4 沒有人有相同的寵物,抽相同牌子的香菸,喝相同的飲料 1 英國人住在紅房子裡 2 瑞典人養了一條狗 3 丹麥人喝茶 4 綠房子在白房子左邊 5 綠房...

藍橋杯 骰子迷題

題目 思路 這道題剛開始看好像有點難,如果用數學的方式求解的話,是有點不好做,坑就在這裡。但是這是程式設計題,貌似沒有暴力解決不了的問題。重新整理下思路,發現得分概率最大其實就等價於自己的骰子中,各個面的數字大於機械人1的骰子的數量的為x1,大於機械人2的骰子的數量為x2,概率p sum x1 i ...