您好,登錄后才能下訂單哦!
本篇內容介紹了“C語言預編譯的方法”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠學有所成!
預定義符號是系統本身定義的:
FILE 進行編譯的源文件的位置
LINE 文件當前的行號
DATE 文件被編譯的日期
TIME 文件被編譯的時間
STDC 如果編譯器遵循 ASNSI C,其值為1,否者未定義
語法:#define name stuff (用stuff替換name)
#define MAX 100 #define STR "hehe" int main() { int max = MAX; printf("%d\n", max); //輸出100 printf("%s\n",STR); //輸出 hehe return 0; }
#define 機制包括了一個機制,允許把參數替換到文本中,這種實現通常稱為宏或者宏定義
宏的申明方式:#define name(parament-list) stuff 其中的parament-list是一個由逗號隔開的符號表,他們可能出現在stuff中。
注意:參數列表的左括號必須與name緊鄰,如果兩者之間有任何空白存在,參數列表就會解釋為stuff 的一部分。
#define SQUARE(X) X*X int main() { int ret = SQUARE(5); printf("%d\n",ret); //輸出25 return 0; }
上面的宏定義代碼存在一定的問題: 如果我們換一個參數(將5換成5+1)輸出的不是36而是11為什呢?
#define SQUARE(X) X*X int main() { int ret = SQUARE(5+1);//替換之后就是(5+1*5+1 = 11) printf("%d\n",ret);//輸出11 return 0; }
沒加括號
因此,用于對數值表達式進行求值的宏定義都應該用這種方式加上括號,避免在使用宏時由于參數中的操作符或臨近操作符之間不可預料的相互作用。
正確的代碼:
#define SQUARE(X) (X)*(X)) int main() { int ret = SQUARE(5+1); printf("%d\n",ret);//輸出36 return 0; }
一, #define NAME “lisa”
程序中有"NAME",但”“內的東西不會被宏替換。
二,宏定義前面的那個必須是合法的用戶標識符
三,宏定義也不是說后面東西隨便寫,不能把字符串的兩個”“拆開。
四: #define NAME “lisa”
程序中有上面的宏定義,并且,程序里有句:
NAMELIST這樣,不會被替換成"lisa"LIST
五,宏不能出現遞歸
#的作用:把參數插入到字符串中
如果我們想要實現一個代碼:把參數插入到字符串中 用到“#”
這里參數a,b就插入到了字符串中了
##的作用:可以把位于它兩邊的符號合成一個符號,它允許宏定義沖從分離的文本片段創建標識符。
圖中的三句代碼是等價的:
printf(“%d\n”,AGE(lisa,24));
printf(“%d\n”,AGE(lisa##24));
printf(“%d\n”,AGE(lisa24));
函數和宏都能實現求兩個數的最大值
//函數 int Max(int x, int y) { return (x > y ? x : y); } //宏 #define MAX(X,Y) ((X)>(Y)?(X):(Y)) int main() { int a = 10; int b = 20; int max = Max(a, b); //輸出20 printf("%d\n",max); max = MAX(a, b); printf("%d\n", max); //輸出20 return 0; }
通過分析上面的代碼實現用宏比用函數會更好,有兩個原因:
用于調用函數和從函數返回的代碼可能比實際執行的這個小型計算工作所需要的時間更多,所以宏比函數在程序的規模和速度方面更勝一籌。
函數的參數必須申明為特定的類型。所以函數只能在類型合適的表達時式上使用。反之宏是與類型無關的。
當然宏相比函數也有劣勢的地方:
每次使用宏的時候,一份宏定義的代碼將替換插入到程序中。除非宏比較短,否者可能大幅度增加程序的長度。
宏沒法調試
宏由于類型無關,也就不夠嚴謹
宏可能會帶來運算符優先級的問題,導致程序容易出錯。
#define定義宏和函數的對比表格
屬性 | #define定義宏 | 函數 |
---|---|---|
代碼長度 | 每次使用時宏代碼都會被插入到程序中除了非常小的宏之外,程序的長度會大幅度增長 | 函數的代碼只出現在一個地方,每次使用這個函數時,都調用那個地方的同一份代碼 |
執行速度 | 更快 | 存在函數的調用和函數的額外開銷,所以速度相對慢一些 |
操作符優先級 | 宏參數的求值是在所有周圍表達式的上下文環境里,除非加上括號,否者鄰近操作符的優先級可能產生不可預料的結果,所以建議宏在書寫的時候多用括號 | 函數參數只在函數調用的時候求值一次,它的結果值傳遞給函數。表達式的求值結果更容易預測。 |
帶有副作用的參數 | 參數可能被替換帶宏中的多個位置,所以帶有副作用的參數求值可能會產生不可預料的結果 | 函數參數只在傳參的時候求值一次,結果更容易控制 |
參數類型 | 宏的參數與類型無關,只要對參數的操作是合法的,它就可以適用于任何參數類型 | 函數的參數與類型有關,如果參數的類型不同,就需要不同的函數,即使他們執行任務是不同的 |
調試和遞歸 | 宏不方便調試,不能遞歸 | 函數可以逐語句調試,可以遞歸 |
命名約定:把宏名全部大寫,函數名不要全部大寫。
#undef 指令用于移除一個宏定義
當#undef 移除宏定義,再次使用報錯。如圖 :
“C語言預編譯的方法”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業相關的知識可以關注億速云網站,小編將為大家輸出更多高質量的實用文章!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。