1. 程式人生 > 其它 >解析 C 語言程式設計中的幾點“坑”操作

解析 C 語言程式設計中的幾點“坑”操作

技術標籤:字串程式語言c語言c++11遊戲

點選上方“逆鋒起筆”,公眾號回覆pdf
領取大佬們推薦的學習資料

編排 | strongerHuang

微信公眾號|嵌入式專欄

單純的程式設計語法可能看起來很簡單,但在實際程式設計中,總會出現各種意想不到的bug,今天給大家分享幾個點C語言中坑人的操作。

1

帶引數的巨集展開順序

#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;}

執行結果:

12f(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)。

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。

3

溢位問題

程式一:

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

執行結果:

死迴圈

淺析:

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

程式二:

#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。關注公眾號 逆鋒起筆,回覆 pdf,下載你需要的各種學習資料。

4

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這一行時會發生記憶體非法訪問導致程式崩潰。

來源:

https://www.cnblogs.com/jungzhang/p/5547348.html

免責宣告:本文素材來源網路,版權歸原作者所有。如涉及作品版權問題,請與我聯絡刪除。

C 語言為什麼不會過時?

C語言 和 C++學習資料分享

只有大佬才能懂:輸出 Hello World 混亂 C 語言的原始碼


點個『在看』支援下

點選“閱讀原文”檢視更多分享,歡迎點分享、收藏、點贊、在看。