1.簡(jiǎn)單的宏定義
#define 標(biāo)識(shí)符 替換列表(替換列表可以是數(shù),字符串字面量,標(biāo)點(diǎn)符號(hào),運(yùn)算符,標(biāo)識(shí)符,關(guān)鍵字,字符常量。注意:替換列表是可以為空的)
典型錯(cuò)誤:
#define N = 100
int a[N]; /*這樣會(huì)成為int a[= 100],這里會(huì)處理成為一種標(biāo)識(shí)記號(hào)一樣*/
#define N 100;
int a[N]; /*帶分號(hào)的定義會(huì)成為int a[100;],這是一種很常見的錯(cuò)誤*/
#define pin (int*);
pin a,b;
int* a,b; /*本意是a和b都是int型指針,但是實(shí)際上變成int* a,b;a是int型指針,而b是int型變量。這是應(yīng)該使用typedef來(lái)代替define,這樣a和b就都是int型指針了。*/
典型的使用方法:
使用宏定義我們可以自己根據(jù)自己的習(xí)慣來(lái)定義甚至改變C語(yǔ)言的語(yǔ)法習(xí)慣,例如:
#define BEGIN {
#define END }
int main()BEGIN
printf ("DEFINE----\n");
END
定義一個(gè)循環(huán)
#define LOOP for(;;)
重新定義數(shù)據(jù)類型
#define IT int
2.帶參數(shù)的宏
#define 標(biāo)識(shí)符(x1,x2,x3...xn) 替換列表 (注意:x1,x2,x3..是宏的參數(shù),標(biāo)識(shí)符和其左括弧之間不能有空格)
使用方法:
#define MAX(x,y) ((x)>(y)?(x):(y))
i=MAX(j+k,m-n);
替換為:
i=MAX((j+k)>(m-n)?(j+k):(m-n));
#define SI_EX(n) ((n)%2==0)
if(SI_EX(i)) i++;
替換為:
if(SI_EX((i)%2==0))
通過以上例子我們可以看出,標(biāo)識(shí)符帶參數(shù)(X1,X2,X3.....)在替換時(shí)會(huì)被替換列表(Y1,Y2,Y3....)對(duì)應(yīng)的替換,但是和順序無(wú)關(guān)。
3.宏的特殊單行定義
#define A(x) T_##x
#define B(x) #@x
#define C(x) #x
x=1
替換為:
A(1)------> T_1 (T_##x,##符號(hào)是記號(hào)粘黏符號(hào),將前后的字符粘黏起來(lái)。)
B(1)------> '1' ( #@x , #@ 符號(hào)會(huì)將宏的參數(shù)進(jìn)行字符串字面量化,并且加‘’號(hào))
C(1)------> "1" ( #x ,#符號(hào)會(huì)將宏的參數(shù)進(jìn)行字符串字面量化,并且加""號(hào))
4.define的多行定義
define可以替代多行的代碼,在每一個(gè)換行的時(shí)候加上一個(gè)"\"
#define MAX(X,Y) do { \
語(yǔ)句1; \
語(yǔ)句2; \
/* 注釋的寫法 */ \
} while(0) /* (no trailing ; ) */ \
5.在大規(guī)模的開發(fā)過程中,特別是跨平臺(tái)和系統(tǒng)的軟件里,define最重要的功能是條件編譯。
#ifdef WINDOWS
......
......
#endif
#ifdef LINUX
......
......
#endif
可以在編譯的時(shí)候通過#define設(shè)置編譯環(huán)境
6.取消宏
#undef 標(biāo)識(shí)符
7.條件編譯
#ifdef XXX…(#else) … #endif
8.預(yù)定義宏
在C語(yǔ)言中收錄了一些有用的宏,這些宏是提供當(dāng)前編譯信息的。
__LINE__ 被編譯文件的行數(shù)
(整型)
__FILE__ 被編譯文件的名字 (字符型)
__DATE__ 編譯日期 (字符型)
__TIME__ 編譯時(shí)間 (字符型)
__STDC__ 如果編譯器接受標(biāo)準(zhǔn)C,那么值為1. (整型)
通常我們?cè)诰幾g的時(shí)候我們可以用這些預(yù)定義的宏來(lái)進(jìn)行排錯(cuò),例如:
一般情況下,當(dāng)出現(xiàn)除零的情況時(shí)編譯器是沒有提示的,我們可以自定義一個(gè)解決辦法。
#define CHECKZERO(divisor)\
if(divisor==0)\
printf("***ATTEMPT TO DIVIDE BY ZERO IN LINE %d of file %s ***\n",__LINE__,__FILE__);\
引用方法
CHECKZERO(j);
k=i/j;