天天看點

C語言的謎題 原文http://coolshell.cn/articles/945.html

這幾天,本站推出了幾篇關于c語言的很多文章如下所示:

我們可以看到很多c語言相關的一些東西。比如《語言的歧義》主要告訴了大家c語言中你意想不到的錯誤以及一些歧義上的東西。而《誰說c語言很簡單》則通過一些看似你從來不可能寫出的代碼來告訴大家c語言并不是一件容易事情。《6個變态的hello world》和《如何弄亂c的源代碼》則以一種極端的方式告訴大家,不要以為咱們自己寫不出混亂的代碼,每個程式員其實都有把代碼搞得一團亂的潛質。通過這些文章,相信你對程式設計或是你覺得很簡單的c語言有了一些了解。是的,很不容易吧,以前是不是低估了程式設計和c語言?今天是否我們又在低估c++和java呢?

本篇文章《c語言的謎題》展示了14個c語言的迷題以及答案,代碼應該是足夠清楚的,而且我也相信有相當的一些例子可能是我們日常工作可能會見得到的。通過這些迷題,希望你能更了解c語言。如果你不看答案,不知道是否有把握回答各個謎題?讓我們來試試。

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

1

2

3

4

5

6

7

8

9

10

11

12

<code>#include &lt;stdio.h&gt;</code>

<code>#include &lt;unistd.h&gt;</code>

<code>int</code> <code>main()</code>

<code>{</code>

<code>    </code><code>while</code><code>(1)</code>

<code>    </code><code>{</code>

<code>        </code><code>fprintf</code><code>(stdout,</code><code>"hello-std-out"</code><code>);</code>

<code>        </code><code>fprintf</code><code>(stderr,</code><code>"hello-std-err"</code><code>);</code>

<code>        </code><code>sleep(1);</code>

<code>    </code><code>}</code>

<code>    </code><code>return</code> <code>0;</code>

<code>}</code>

參考答案:stdout和stderr是不是同裝置描述符。stdout是塊裝置,stderr則不是。對于塊裝置,隻有當下面幾種情況下才會被輸入,1)遇到回車,2)緩沖區滿,3)flush被調用。而stderr則不會。

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

<code>    </code><code>int</code> <code>a = 1,2;</code>

<code>    </code><code>printf</code><code>(</code><code>"a : %d\n"</code><code>,a);</code>

參考答案:這個程式會得到編譯出錯(文法出錯),逗号表達式是沒錯,可是在初始化和變量聲明時,逗号并不是逗号表達式的意義。這點要區分,要修改上面這個程式,你需要加上括号: int a = (1,2);

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

<code>    </code><code>int</code> <code>i=43;</code>

<code>    </code><code>printf</code><code>(</code><code>"%d\n"</code><code>,</code><code>printf</code><code>(</code><code>"%d"</code><code>,</code><code>printf</code><code>(</code><code>"%d"</code><code>,i)));</code>

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

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

<code>    </code><code>float</code> <code>a = 12.5;</code>

<code>    </code><code>printf</code><code>(</code><code>"%d\n"</code><code>, a);</code>

<code>    </code><code>printf</code><code>(</code><code>"%d\n"</code><code>, (</code><code>int</code><code>)a);</code>

<code>    </code><code>printf</code><code>(</code><code>"%d\n"</code><code>, *(</code><code>int</code> <code>*)&amp;a);</code>

參考答案:

該項程式輸出如下所示,

1095237632

原因是:浮點數是4個位元組,12.5f 轉成二進制是:01000001010010000000000000000000,十六進制是:0×41480000,十進制是:1095237632。是以,第二和第三個輸出相信大家也知道是為什麼了。而對于第一個,為什麼會輸出0,我們需要了解一下float和double的記憶體布局,如下:

float: 1位符号位(s)、8位指數(e),23位尾數(m,共32位)

double: 1位符号位(s)、11位指數(e),52位尾數(m,共64位)

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

float版:0×41480000 (在記憶體中是:00 00 48 41)

double版:0×4029000000000000 (在記憶體中是:00 00 00 00 00 00 29 40)

而我們的%d要求是一個4位元組的int,對于double的記憶體布局,我們可以看到前四個位元組是00,是以輸出自然是0了。

這個示例向我們說明printf并不是類型安全的,這就是為什麼c++要引如cout的原因了。

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

file1.c

<code>int</code> <code>arr[80];</code>

file2.c

<code>extern</code> <code>int</code> <code>*arr;</code>

<code>    </code><code>arr[1] = 100;</code>

<code>    </code><code>printf</code><code>(</code><code>"%d\n"</code><code>, arr[1]);</code>

參考答案:該程式可以編譯通過,但運作時會出錯。為什麼呢?原因是,在另一個檔案中用 extern int *arr來外部聲明一個數組并不能得到實際的期望值,因為他們的類型并不比對。是以導緻指針實際并沒有指向那個數組。注意:一個指向數組的指針,并不等于一個數組。修改:extern int arr[]。(參考:iso c語言 6.5.4.2 節)

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

13

14

15

16

<code>    </code><code>int</code> <code>a=1;</code>

<code>    </code><code>switch</code><code>(a)</code>

<code>        </code><code>int</code> <code>b=20;</code>

<code>        </code><code>case</code> <code>1:</code>

<code>            </code><code>printf</code><code>(</code><code>"b is %d\n"</code><code>,b);</code>

<code>            </code><code>break</code><code>;</code>

<code>        </code><code>default</code><code>:</code>

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

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

<code>    </code><code>char</code> <code>str[80];</code>

<code>    </code><code>printf</code><code>(</code><code>"enter the string:"</code><code>);</code>

<code>    </code><code>scanf</code><code>(</code><code>"%s"</code><code>,str);</code>

<code>    </code><code>printf</code><code>(</code><code>"you entered:%s\n"</code><code>,str);</code>

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

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

<code>    </code><code>int</code> <code>i;</code>

<code>    </code><code>i = 10;</code>

<code>    </code><code>printf</code><code>(</code><code>"i : %d\n"</code><code>,i);</code>

<code>    </code><code>printf</code><code>(</code><code>"sizeof(i++) is: %d\n"</code><code>,</code><code>sizeof</code><code>(i++));</code>

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

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

17

18

19

20

21

22

<code>#include &lt;stdlib.h&gt;</code>

<code>#define sizeof(arr) (sizeof(arr)/sizeof(arr[0]))</code>

<code>#define printint(expr) printf("%s:%d\n",#expr,(expr))</code>

<code>    </code><code>/* the powers of 10 */</code>

<code>    </code><code>int</code> <code>pot[] = {</code>

<code>                    </code><code>0001,</code>

<code>                    </code><code>0010,</code>

<code>                    </code><code>0100,</code>

<code>                    </code><code>1000</code>

<code>                </code><code>};</code>

<code>    </code><code>for</code><code>(i=0;i&lt;sizeof(pot);i++)</code>

<code>        </code><code>printint(pot[i]);</code>

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

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

11、下面的輸出是什麼?

<code>    </code><code>int</code> <code>i = 6;</code>

<code>    </code><code>if</code><code>( ((++i &lt; 7) &amp;&amp; ( i++/6)) || (++i &lt;= 9))</code>

<code>        </code><code>;</code>

<code>    </code><code>printf</code><code>(</code><code>"%d\n"</code><code>,i);</code>

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

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

<code>    </code><code>int</code> <code>a=3, b = 5;</code>

<code>    </code><code>printf</code><code>(&amp;a[</code><code>"ya!hello! how is this? %s\n"</code><code>], &amp;b[</code><code>"junk/super"</code><code>]);</code>

<code>    </code><code>printf</code><code>(&amp;a[</code><code>"what%c%c%c  %c%c  %c !\n"</code><code>], 1[</code><code>"this"</code><code>],</code>

<code>        </code><code>2[</code><code>"beauty"</code><code>],0[</code><code>"tool"</code><code>],0[</code><code>"is"</code><code>],3[</code><code>"sensitive"</code><code>],4[</code><code>"cccccc"</code><code>]);</code>

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

hello! how is this? super that is c !

本例主要展示了一種另類的用法。下面的兩種用法是相同的:

“hello”[2] 2["hello"]

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

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

<code>    </code><code>char</code> <code>dummy[80];</code>

<code>    </code><code>printf</code><code>(</code><code>"enter a string:\n"</code><code>);</code>

<code>    </code><code>scanf</code><code>(</code><code>"%[^r]"</code><code>,dummy);</code>

<code>    </code><code>printf</code><code>(</code><code>"%s\n"</code><code>,dummy);</code>

參考答案:本例的輸出是“hello, wo”,scanf中的”%[^r]“是從中作梗的東西。意思是遇到字元r就結束了。

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

<code>#define printint(expr) printf("%s : %d\n",#expr,(expr))</code>

<code>int</code> <code>fivetimes(</code><code>int</code> <code>a)</code>

<code>    </code><code>int</code> <code>t;</code>

<code>    </code><code>t = a&lt;&lt;2 + a;</code>

<code>    </code><code>return</code> <code>t;</code>

<code>    </code><code>int</code> <code>a = 1, b = 2,c = 3;</code>

<code>    </code><code>printint(fivetimes(a));</code>

<code>    </code><code>printint(fivetimes(b));</code>

<code>    </code><code>printint(fivetimes(c));</code>

參考答案:本題的問題在于函數fivetimes中的表達式“t = a&lt;&lt;2 + a;”,對于a&lt;&lt;2這個位操作,優先級要比加法要低,是以這個表達式就成了“t = a &lt;&lt; (2+a)”,于是我們就得不到我們想要的值。該程式修正如下:

<code>    </code><code>t = (a&lt;&lt;2) + a;</code>

(全文完)