總結幾個C語言中的“坑”

今天給大家分享幾個C語言中的坑。
一、帶參數的宏展開順序

#include <stdio.h>
#define f(a,b) a##b
#define g(a) #a
#define h(a) g(a)

int main(void)
{
     printf("%s\n",h(f(1,2)));
     printf("%s\n",g(f(1,2)));

     return 0;
}

運行結果:

12

f(1,2)

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

所以printf("%s\n",h(f(1,2)));這條語句的展開順序爲:h(f(1,2))(沒有#) —>> f(1,2)(到達最裏層依然沒有#) ---->> h(12) ---->> 12。

然而printf("%s\n",g(f(1,2)));這條語句的展開順序是:g(f(1,2))(碰到#即刻結束探尋,開始展開) ----->>f(1,2)。
二、類型轉換

#include <stdio.h>
int main(void)
{
int a = -10;
unsigned b = 1;
if(a+b > 0)
printf("a + b > 0\n");
else if (a + b < 0)
printf("a + b < 0\n");
else
printf("a + b = 0\n");
return 0;
}

運行結果:

a + b > 0

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

三、溢出問題

程序一:

unsigned i;
for (i = 110; i >= 0; i--)
  printf("%u\n",i);

運行結果:
死循環

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

其實做爲一個學習者,有一個學習的氛圍跟一個交流圈子特別重要這裏我推薦一個C語言C++交流羣1075673198,不管你是小白還是轉行人士歡迎入駐,大家一起交流成長。
程序二:

#include <stdio.h>
#include <string.h>
int main(void)
{
  char a[1000];
  int i;
  for(i = 0; i < 1000; i++)
    a[i] = -1 -  i;
  printf("%d\n",strlen(a));
  return 0;
}

運行結果:
255

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

其實做爲一個學習者,有一個學習的氛圍跟一個交流圈子特別重要這裏我推薦一個C語言C++交流羣1075673198,不管你是小白還是轉行人士歡迎入駐,大家一起交流成長。

四、strcpy函數

void test()
{
char str[10],str1[10];
int i;
for(i = 0; i < 10; i++)
  {
    str1[i] = 'a' + i;
  }
strcpy(str,str1);
}

淺析:
這段代碼第一眼看過去是沒問題的,但是再看一眼就能夠很輕鬆找到錯誤了,strcpy函數是拷貝字符串的函數,它是以’\0’爲結尾的,因此當程序運行strcpy這一行時會發生內存非法訪問導致程序崩潰。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章