宏定義

宏定義

宏定義,別名宏代換,是C提供的三種預處理功能的其中一種,這三種預處理包括:宏定義、檔案包含、條件編譯。宏定義又稱為宏代換、宏替換,簡稱“宏”。預處理(預編譯)工作也叫做宏展開:將宏名替換為字元串。一切以換為前提、做任何事情之前先要換,準確理解之前就要“換”。NAMELIST整體是個標識符,而沒有NAME標識符,所以不替換。實參是什麼,都不會影響其被替換成"a"的命運。STR(A,B)如果實參過多,則編譯器會把多餘的參數捨去。

宏定義

宏定義是C提供的三種預處理功能的其中一種,這三種預處理包括:宏定義、檔案包含、

條件編譯

參數

不帶參數

宏定義宏定義

宏定義又稱為宏代換、宏替換,簡稱“宏”。

格式:

#define 標識符 字元串

其中的標識符就是所謂的符號常量,也稱為“宏名”。

預處理(預編譯)工作也叫做宏展開:將宏名替換為字元串。

掌握"宏"概念的關鍵是“換”。一切以換為前提、做任何事情之前先要換,準確理解之前就要“換”。

即在對相關命令或語句的含義和功能作具體分析之前就要換:

例:

#define Pi 3.1415926

把程式中出現的Pi全部換成3.1415926

說明:

(1)宏名一般用大寫

(2)使用宏可提高程式的通用性和易讀性,減少不一致性,減少輸入錯誤和便於修改。例如:數組大小常用宏定義

(3)預處理是在編譯之前的處理,而編譯工作的任務之一就是語法檢查,預處理不做語法檢查。

(4)宏定義末尾不加分號;

(5)宏定義寫在函式的花括弧外邊,作用域為其後的程式,通常在檔案的最開頭。

(6)可以用#undef命令終止宏定義的作用域

(7)宏定義允許嵌套

(8)字元串( " " )中永遠不包含宏

(9)宏定義不分配記憶體,變數定義分配記憶體。

(10)宏定義不存在類型問題,它的參數也是無類型的。

帶參數

除了一般的字元串替換,還要做參數代換

格式:

#define宏名(參數表) 字元串

例如:#define S(a,b) a*b

area=S(3,2);第一步被換為area=a*b; ,第二步被換為area=3*2;

類似於函式調用,有一個啞實結合的過程:

(1)實參如果是表達式容易出問題

#define S(r) r*r

area=S(a+b);第一步換為area=r*r;,第二步被換為area=a+b*a+b;

正確的宏定義是#define S(r) ((r)*(r))

(2)宏名和參數的括弧間不能有空格

(3)宏替換隻作替換,不做計算,不做表達式求解

(4)函式調用在編譯後程式運行時進行,並且分配記憶體。宏替換在編譯前進行,不分配記憶體

(5)宏的啞實結合不存在類型,也沒有類型轉換。

(6)宏展開使源程式變長,函式調用不會

(7)宏展開不占運行時間,只占編譯時間,函式調用占運行時間(分配記憶體、保留現場、值傳遞、返回值)

宏定義其他冷門、重點知識

#define用法

1、 用無參宏定義一個簡單的

常量

#define LEN 12

這個是最常見的用法,但也會出錯。

比如下面幾個知識點你會嗎?可以看下:

(1) #define NAME "zhangyuncong"

程式中有"NAME"則,它會不會被替換呢?

(2) #define 0x abcd

可以嗎?也就是說,可不可以用把

標識符的字母替換成別的東西?

(3) #define NAME "zhang

這個可以嗎?

(4) #define NAME "zhangyuncong"

程式中有上面的宏定義,並且,程式里有句:

NAMELIST這樣,會不會被替換成"zhangyuncong"LIST

四個題答案都是十分明確的。

第一個,""內的東西不會被宏替換。這一點應該大都知道。

第二個,宏定義前面的那個必須是合法的

用戶標識符

第三個,宏定義也不是說後面東西隨便寫,不能把字元串的兩個""拆開。

第四個:只替換標識符,不替換別的東西。NAMELIST整體是個標識符,而沒有NAME標識符,所以不替換。

也就是說,這種情況下記住:#define 第一位置第二位置

(1) 不替換程式中字元串里的東西。

(2) 第一位置只能是合法的標識符(可以是關鍵字)

(3) 第二位置如果有字元串,必須把""配對。

(4) 只替換與第一位置完全相同的標識符

還有就是老生常談的話:記住這是簡單的替換而已,不要在中間計算結果,一定要替換出

表達式之後再算。

2、 帶參宏一般用法

比如#define MAX(a,b) ((a)>(b)?(a):(b))

則遇到MAX(1+2,value)則會把它替換成:

((1+2)>(value)?(1+2):(value))

注意事項和無參宏差不多。

但還是應注意

#define FUN(a) "a"

則,輸入FUN(345)會被替換成什麼?

其實,如果這么寫,無論宏的

實參是什麼,都不會影響其被替換成"a"的命運。

也就是說,""內的字元不被當成

形參,即使它和一模一樣。

那么,你會問了,我要是想讓這裡輸入FUN(345)它就替換成"345"該怎么實現呢?

請看下面關於#的用法

3、 有參宏定義中#的用法

#define STR(str) #str

#用於把宏定義中的參數兩端加上字元串的""

比如,這裡STR(my#name)會被替換成"my#name"

一般由任意

字元都可以做形參,但以下情況會出錯:

STR())這樣,

編譯器不會把“)”當成STR()的參數。

STR(,)同上,編譯器不會把“,”當成STR的參數。

STR(A,B)如果實參過多,則編譯器會把多餘的參數捨去。(VC++2008為例)

STR((A,B))會被解讀為實參為:(A,B),而不是被解讀為兩個實參,第一個是(A第二個是B)。

4、 有參宏定義中##的用法

#define WIDE(str) L##str

則會將形參str的前面加上L

比如:WIDE("abc")就會被替換成L"abc"

如果有#define FUN(a,b) vo##a##b()

那么FUN(id ma,in)會被替換成void main()

5、 多行宏定義:

#define doit(m,n) for(int i=0;i<(n);++i)\

{\

m+=i;\

}

相關詞條

相關搜尋

熱門詞條

聯絡我們