宏定義了一個(gè)代表特定內容的標識符。
預處理過(guò)程會(huì )把源代碼中出現的宏標識符替換成宏定義時(shí)的值。
宏最常見(jiàn)的用法是定義代表某個(gè)值的全局符號。
宏的第二種用法是定義帶參數的宏,這樣的宏可以象函數一樣被調用,但它是在調用語(yǔ)句處展開(kāi)宏,并用調用時(shí)的實(shí)際參數來(lái)代替定義中的形式參數。
1.#define指令
#define MAX_NUM 10
int array[MAX_NUM];
for(i=0;i
#define VERSION "Version 1.0 Copyright(c) 2003"
2.帶參數的#define指令
#define IS_EVEN(n) ((n)%2==0)
#define MAX(x,y) ((x)>(y) ? (x) :(y))
#define Cube(x) (x)*(x)*(x)
可以是任何數字表達式甚至函數調用來(lái)代替參數x。
3.#運算符
#的功能是將其后面的宏參數進(jìn)行字符串化操作(Stringfication),簡(jiǎn)單說(shuō)就是在對它所引用的宏變量通過(guò)替換后在其左右各加上一個(gè)雙引號。例如:
#define _STR(s) #s
#define WARN_IF(EXP) \
do{ if (EXP) \
fprintf(stderr, "Warning: " #EXP "\n"); } \
while(0)
那么實(shí)際使用中會(huì )出現下面所示的替換過(guò)程:
WARN_IF (divider == 0);
被替換為
do
{
if (divider == 0)
fprintf(stderr, "Warning" "divider == 0" "\n");
} while(0);
這樣每次divider(除數)為0的時(shí)候便會(huì )在標準錯誤流上輸出一個(gè)提示信息。
再例如下面的例子:
#define FILL(a) {a, #a}
enum IDD{OPEN, CLOSE};
typedef struct MSG{
IDD id;
const char * msg;
}MSG;
MSG _msg[] = {FILL(OPEN), FILL(CLOSE)};
相當于:
MSG _msg[] = {{OPEN, "OPEN"},
{CLOSE, "CLOSE"}};
4.##運算符
##運算符用于把參數連接到一起。
預處理程序把出現在##兩側的參數合并成一個(gè)符號。
看下面的例子:
#define NUM(a,b,c) a##b##c
#define STR(a,b,c) a##b##c
main()
{
printf("%d\n",NUM(1,2,3));
printf("%s\n",STR("aa","bb","cc"));
}
最后程序的輸出為:
123
aabbcc
再看下面的例子:
struct command
{
char * name;
void (*function) (void);
};
#define COMMAND(NAME) { NAME, NAME ## _command }
// 然后你就用一些預先定義好的命令來(lái)方便的初始化一個(gè)command結構的數組了:
struct command commands[] = {
COMMAND(quit),
COMMAND(help),
...
}
COMMAND宏在這里充當一個(gè)代碼生成器的作用,這樣可以在一定程度上減少代碼密度,間接地也可以減少不留心所造成的錯誤。我們還可以n個(gè)##符號連接 n+1個(gè)Token。比如:
#define LINK_MULTIPLE(a,b,c,d) a##_##b##_##c##_##d
typedef struct _record_type LINK_MULTIPLE(name,company,position,salary);
// 這里這個(gè)語(yǔ)句將展開(kāi)為:
// typedef struct _record_type name_company_position_salary;
5.特殊的宏
#error指令將使編譯器顯示一條錯誤信息,然后停止編譯。
#line指令可以改變編譯器用來(lái)指出警告和錯誤信息的文件號和行號。
#pragma指令沒(méi)有正式的定義。編譯器可以自定義其用途。典型的用法是禁止或允許某些煩人的警告信息。
...在C宏中稱(chēng)為Variadic Macro,也就是變參宏。
Compiled on Dec 23 1996 at 22:18:48
6.預定義宏
__LINE__ 被編譯的文件的行數
__FILE__ 被編譯的文件的名字
__DATE__ 編譯的日期(格式"Mmm dd yyyy")
__TIME__ 編譯的時(shí)間(格式"hh:mm:ss")
__STDC__ 如果編譯器接受標準C,那么值為1
printf("Compiled on %s at %s\n", __DATE__, __TIME__);
每次程序開(kāi)始執行,程序都會(huì )顯示,用于確認版本:
Compiled on Dec 23 1996 at 22:18:48
下面的宏可以幫助我們查明錯誤的根源:
#define CHECK_ZERO(divisor) \
if (divisor == 0) \
printf("*** Attempt to divide by zero on line %d " \
"of file %s ***\n",__LINE__, __FILE__)
CHECK_ZERO宏應該在除法運算前被調用:
CHECK_ZERO(j);k = i / j;
如果j是0,會(huì )顯示出如下形式的信息:
*** Attempt to divide by zero on line 9 of file FOO.c ***
通用的、用于錯誤檢測的宏——assert宏;
7.注意
宏名和參數的括號間不能有空格;
宏替換只作替換,不做計算,不做表達式求解;
函數調用在編譯后程序運行時(shí)進(jìn)行,并且分配內存。宏替換在編譯前進(jìn)行,不分配內存;
函數只有一個(gè)返回值,利用宏則可以設法得到多個(gè)值;
宏展開(kāi)使源程序變長(cháng),函數調用不會(huì );
宏展開(kāi)不占運行時(shí)間,只占編譯時(shí)間,函數調用占運行時(shí)間(分配內存、保留現場(chǎng)、值傳遞、返回值);
使用條件編譯可以使目標程序變小,運行時(shí)間變短;
預編譯使問(wèn)題或算法的解決方案增多,有助于我們選擇合適的解決方案。
聯(lián)系客服