【總結】 幾個C語言中的「坑」

  • 2019 年 10 月 11 日
  • 筆記

1、帶參數的宏展開順序

運行結果

12

f(1,2)

分析

本題中的#運算符可以利用宏參數創建字元串。##運算符和#運算符一樣也可以用於類函數宏的替換部分。另外,##還可以用於類對象宏的替換部分,這個運算符可以把兩個語言符號組合成單個語言符號,所以該運算符也被成為「預處理粘合劑」。類參數宏展開遵循一定的順序,先從外層開始探尋如果遇到#即刻結束探尋,從遇到#處開始一步一步向外層展開,如果沒有遇到#探尋到最裡層結束探尋,然後一步一步向外層展開。

所以printf("%sn",h(f(1,2)));這條語句的展開順序為:h(f(1,2))(沒有#) —>> f(1,2)(到達最裡層依然沒有#) —->> h(12) —->> 12。然而printf("%sn",g(f(1,2)));這條語句的展開順序是:g(f(1,2))(碰到#即刻結束探尋,開始展開) —–>>f(1,2)

2、類型轉換

運行結果

a + b > 0

分析

第一眼看到這道題心裡想到這不明擺著 -10 + 1 < 0么,如此easy的題目還要算嗎?當程式運行出結果時頓時傻眼了,仔細看了看數據類型發現問題出在了類型的轉換上。眾所周知,在不同類型的數據進行運算時如果不進行特別的轉換那麼在數據運算時會先將表示範圍較小的數據自動轉換成表示範圍更廣的數,再參與運算,所以本題中會先將int型的a轉換成unsigned int型,通過補碼運算得知該值為:4294967286,該值加上1會肯定會遠大於0,因此輸出的是a+b>0。

3、溢出問題

程式1

運行結果

死循環

分析

該題的坑就在於沒有注意到unsigned int 的存儲範圍,當小於零溢出時又會從unsigned int 的最大值開始遞減,這就彷彿進入了一個圓環,永遠都沒有辦法找到跳出圓環形跑道的缺口。

程式2

運行結果

255

分析

這道題看上去很簡單但是卻暗藏殺機,很少有人能夠答對,當i從0開始自增,自增到127時-1 – 127 = -128,而這個數正好是char型變數所能表示的最小數字,i再自增一次就會溢出,變成char所能表示的最大數字,這樣又進入了上一題的那個「環」,當i增加到255時-1 – i = 0,此時第一次出現了0,而strlen函數碰到''就結束(不包括),因此輸出結果為255。

4、float變數與"零"值的比較

眾所周知float型和double型都是有精度限制的,在電腦中用一個近似值來表示任意某個實數。具體的說,這個實數由一個整數或定點數(即尾數)乘以某個基數(電腦中通常是2)的整數次冪得到,因此使用 if(a == 0) 這樣的語句往往會發生意想不到的錯誤,那麼怎樣才能比較精確的將浮點型變數與「零」值進行比較呢?這裡給出了一種常用的方法。

注意

EPSINON是一個宏,它的值可以改成你想要的精度,通過這種方式也可以進行其他任何值的精度比較

由於浮點型變數的存儲機制最好不要進行很大的浮點數與很小的浮點數之間進行運算,

比如11111111111.000+0.00000000000001這個運算結果可能會很奇葩

5、strcpy函數運行機制的理解

(來自華為面試題,程式本意是想實現字元串複製)

注意

這段程式碼第一眼看過去是沒問題的,但是再看一眼就能夠很輕鬆找到錯誤了,strcpy函數是拷貝字元串的函數,它是以''為結尾的,因此當程式運行strcpy這一行時會發生記憶體非法訪問導致程式崩潰。