C語言的一些誤用和知識總結
在學習微控制器的時候才真正知道C語言是什麼它是來幹什麼的~但是C語言用到嵌入式只是它小小的一部分他的應用還有很多地方呢,呵呵我們這裡就不討論這個了。我們是不是在寫程式的時候錯誤很多就算編譯通過了也達不到我們預期的結果,完了自己是不是也很難找到錯在哪兒吧?我絕得語言之所以能稱之為語言,它肯定是一種工具一種相互交流相互通訊相互傳達之間的意圖的工具,作為語言那肯定得有自己的語法,要想相互交流肯定得先學好它的語法吧(比如像表示式,函式,迴圈,指標)我稱之為C語言的語法。C語言雖然很強大但是他也有不少陷阱,所以我發這篇部落格有兩個目的一是:把C語言一些誤用易錯的地方總結一下,二是把C語言一些基本語法總結一下~
第一次:
1.關於自增自減(即++i,i++)
要想給一個數加一或減一我們可以:
i += 1;
j -= 1;
而C語言還允許用++和--運算子,其實這裡有誤導,因為++和--可以作為字首和字尾,這樣的話他們就有可能改變運算元的值,下面讓我們來看看:
i = 1;
printf("i is %d\n",++i); /* prints i is 2 */
printf("i is %d\n",i); /* prints i is 2 */
計算表示式i++的結果是i,但是會引發i隨後進行自增:
i = 1;
printf("i is %d\n",i++); /* prints i is 1/ */
printf("i is %d\n",i); /* prints i is 2 */
第一個printf 顯示了i自增前的原始值,第二個printf顯示了i變化後的新值;當然 -- 類似我就不舉例了~
但在同一個表示式中多次使用++和--往往很難理解我們看看下面的例子:
i = 1;
j = 2;
k = ++i + j++;
i,j,k最終值分別是2,3,4而++i是2 j++是2;
總結:不管是++i還是i++執行這條語句後i的值都加一了只是(++i)的值加一了而(i++)沒變,
2.typedef與#define
2.1.typedef
C語言除了直接使用標準的型別名(如 int char float double)和自己宣告的結構體、共用體、指標、列舉型別外,還可以用typedef宣告新的型別名來代替現有的型別名。
typedef unsigned char u8;
typedef unsigned int u16;
u8 count;
u16 time;
typedef struct
{
u8 month;
u8 day;
u16 year;
}DATE;
DATE brithday;
總結一下,宣告新的型別名的方法:
1.先按定義變數的方法寫出定義體(如 unsigned int i)
2.在變數名換成新的變數名(如將 i換成u16)
3.在最前面加上typedef (typedef unsigned int u16)
4.然後用新型別名去定義變數
2.2 #define
2.1.1不帶引數的巨集定義
#define 識別符號 字串
#define PI 3.1415926
注意:
1.它的作用是在本程式中用指定的識別符號PI來代替3.1415926
2.巨集定義是用巨集來代替字串也就是做簡單的置換,不做正確性檢查如果寫成
#define PI 3.l4l6926
即把1寫成了字母l但是預處理照常代入不做任何語法檢查!!
2.1.2帶引數的巨集定義
#define 巨集名(引數) 字串
#define S(a,b) a*b
area = S(a,b);
#define MAX(x,y) (x)>(y) ? (x):(y)
3.typedef和#define的區別
一般來說typedef 因為它能正確處理指標型別
typedef char *String1;
#define String2 char *
String1 s1,s2;
String2 s3,s4;
s1,s2,s3 被定義為了char* 但s4卻被定義為了char型
3. static 變數
static變數大致分為三種用法
1. 用於區域性變數中,成為靜態區域性變數. 靜態區域性變數有兩個用法,記憶功能和全域性生存期.
2. 用於全域性變數,主要作用是限制此全域性變數被其他的檔案呼叫.
3. 用於類中的成員.表示這個成員是屬於這個類但是不屬於類中任意特定物件
1. 靜態區域性變數
靜態區域性變數屬於靜態儲存方式,它具有以下特點:
(1)靜態區域性變數在函式內定義 它的生存期為整個源程式,但是其作用域仍與自動變數相同,只能在定義該變數的函式內使用該變數。退出該函式後, 儘管該變數還繼續存在,但不能使用它。
(2)允許對構造類靜態區域性量賦初值 例如陣列,若未賦以初值,則由系統自動賦以0值。
(3) 對基本型別的靜態區域性變數若在說明時未賦以初值,則系統自動賦予0值。而對自動變數不賦初值,則其值是不定的。 根據靜態區域性變數的特點, 可以看出它是一種生存期為整個源程式的量。雖然離開定義它的函式後不能使用,但如再次呼叫定義它的函式時,它又可繼續使用, 而且儲存了前次被呼叫後留下的值。 因此,當多次呼叫一個函式且要求在呼叫之間保留某些變數的值時,可考慮採用靜態區域性變數。雖然用全域性變數也可以達到上述目的,但全域性變數有時會造成意外的副作用,因此仍以採用區域性靜態變數為宜。
舉例如下:
void fun()
{
static int a = 1;
a++;
}
在第一次進入這個函式的時候,變數a被初始化為1!並接著自增1,以後每次進入該函式,a就不會被再次初始化了,僅進行自增1的操作;在static發明前,要達到同樣的功能,則只能使用全域性變數:
int a = 1;
void fun()
{
a++;
}