三種預處理功能
C++提供的編譯預處理功能主要有以下三種:
(一)宏定義
(二) 檔案包含
(三) 條件編譯
在C++中,我們一般用const定義符號常量。很顯然,用const定義常量比用define定義常量更好。
使用宏定義注意點
在使用宏定義時應注意的是:
(a) 在書寫#define 命令時,注意<宏名>和<字元串>之間用空格分開,而不是用等號連線。
(b) 使用#define定義的標識符不是變數,它只用作宏替換,因此不占有記憶體。
(c) 習慣上用大寫字母表示<宏名>,這只是一種習慣的約定,其目的是為了與變數名區分,因為變數名
通常用小寫字母。
如果某一個標識符被定義為宏名後,在取消該宏定義之前,不允許重新對它進行宏定義。取消宏定義使用如下命令:
#undef<標識符>
其中,undef是關鍵字。該命令的功能是取消對<標識符>已有的宏定義。被取消了宏定義的標識符,可以對它重新進行定義。
宏定義可以嵌套,已被定義的標識符可以用來定義新的標識符。例如:
#define PI 3.14159265
#define R 10
#define AREA (PI*R*R)
帶參數的宏定義形式
帶參數的宏定義的一般形式如下:
#define <宏名>(<參數表>) <宏體>
其中, <宏名>是一個標識符,<參數表>中的參數可以是一個,也可以是多個,視具體情況而定,當有多個參數的時候,每個參數之間用逗號分隔。<宏體>是被替換用的字元串,宏體中的字元串是由參數表中的各個參數組成的表達式。例如:
#define SUB(a,b) a-b
如果在程式中出現如下語句:
result=SUB(2, 3)
則被替換為:
result=2-3;
如果程式中出現如下語句:
result= SUB(x+1, y+2);
則被替換為:
result=x+1-y+2;
在這樣的宏替換過程中,其實只是將參數表中的參數代入到宏體的表達式中去,上述例子中,即是將表達式中的a和b分別用2和3代入。
我們可以發現:帶參的宏定義與函式類似。如果我們把宏定義時出現的參數視為形參,而在程式中引用宏定義時出現的參數視為實參。那么上例中的a和b就是形參,而2和3以及x+1和y+2都為實參。在宏替換時,就是用實參來替換<宏體>中的形參。
使用帶參數宏定義注意點
在使用帶參數的宏定義時需要注意的是:
(1)帶參數的宏定義的<宏體>應寫在一行上,如果需要寫在多行上時,在每行結束時,使用續行符 "\"結
束,並在該符號後按下回車鍵,最後一行除外。
(2)在書寫帶參數的宏定義時,<宏名>與左括弧之間不能出現空格,否則空格右邊的部分都作為宏體。
例如:
#define ADD (x,y) x+y
將會把"(x,y)x+y"的一個整體作為被定義的字元串。
(3)定義帶參數的宏時,宏體中與參數名相同的字元串適當地加上圓括弧是十分重要的,這樣能夠避免
可能產生的錯誤。例如,對於宏定義:
#define SQ(x) x*x
當程式中出現下列語句:
m=SQ(a+b);
替換結果為:
m=a+b*a+b;
這可能不是我們期望的結果,如果需要下面的替換結果:
m=(a+b)*(a+b);
應將宏定義修改為:
#define SQ(x) (x)*(x)
對於帶參的宏定義展開置換的方法是:在程式中如果有帶實參的宏(如"SUB(2,3)"),則按"#define"命令行中指定的字元串從左到右進行置換。如果串中包含宏中的形參(如a、b),則將程式語句中相應的實參(可以是常量、變數或者表達式)代替形參,如果宏定義中的字元串中的字元不是參數字元(如a-b中的-號),則保留。這樣就形成了置換的字元串。
(4) 定義帶參數的宏後,使用時最好避免使用表達式傳參。這樣可以在複雜的宏定義中避免(3)中出現的問題