在程式開始以#開頭的命令,他們是預編譯命令。有三類預編譯命令:巨集定義命令、文件包含命令、條件編譯命令;今天聊聊巨集定義: 巨集定義命令將一個標識符定義為一個字元串,源程式中的該標識符均以指定的字元串來代替。因此預處理命令後通常不加分號。這並不是說所有的預處理命令後都不能有分號出現。由於巨集定義只是用巨集名對 ...
在程式開始以#開頭的命令,他們是預編譯命令。有三類預編譯命令:巨集定義命令、文件包含命令、條件編譯命令;今天聊聊巨集定義: 巨集定義命令將一個標識符定義為一個字元串,源程式中的該標識符均以指定的字元串來代替。因此預處理命令後通常不加分號。這並不是說所有的預處理命令後都不能有分號出現。由於巨集定義只是用巨集名對一個字元串進行簡單的替換,因此如果在巨集定義命令後加了分號,將會連同分號一起進行置換。 在C++中,我們一般用const定義常量。很顯然,用const定義常量比用define定義常量更好。 帶參數的巨集定義的一般形式如下: #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) 在書寫#define 命令時,註意<巨集名>和<字元串>之間用空格分開,而不是用等號連接。 (b) 使用#define定義的標識符不是變數,它只用作巨集替換,因此不占有記憶體。 (c) 習慣上用大寫字母表示<巨集名>,這隻是一種習慣的約定,其目的是為了與變數名區分,因為變數名 通常用小寫字母。 如果某一個標識符被定義為巨集名後,在取消它之前,不允許重新對它進行巨集定義。取消巨集定義使用如下命令: #undef<標識符> 其中,undef是關鍵字。該命令的功能是取消對<標識符>已有的巨集定義。被取消了巨集定義的標識符,可以對它重新進行定義。 巨集定義可以嵌套,已被定義的標識符可以用來定義新的標識符。例如: #define PI 3.14159265#define R 10#define AREA (PI*R*R)註意事項 在使用帶參數的巨集定義時需要註意的是: (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)中出現的問題