宏的使用的核心,就是替換,換是最關鍵的。
1、不帶參數的宏定義
這是最簡單的了,比如#define PI 3.1415926
預編譯的時候,把代碼中的PI替換就行了。一般情況下宏名用大寫字母,不要在行末加分號。
2、帶參數的宏定義
不只是進行宏體的替換,還要進行參數的替換。
比如:#define MAX(x,y) (x>y)?x:y
宏展開的時候要將語句中宏名后面的括號內的實參代替形參。另外為了避免發生錯誤,凡是帶運算符的參數要用圓括號括起來。
3、不常見但是很重要的用法
(1)#define FUN(a) "a"
那么當輸入FUN(345)是,照樣會被替換成“a”,無論宏的實參是什么,都不會影響其被替換成"a"的命運。
也就是說,""內的字符不被當成形參,即使它和一模一樣。
(2)有參宏定義中#的用法
#define STR(str) #str
str前面的那個#用于把宏定義中的參數兩端加上字符串的""
比如代碼中有STR(my#name),那么在展開的時候被替換成"my#name"。
一般由任意字符都可以做形參,但以下情況會出錯:
STR())這樣,編譯器不會把“)”當成STR()的參數。
STR(,)同上,編譯器不會把“,”當成STR的參數。
STR(A,B)如果實參過多,則編譯器會把多余的參數舍去。(VC++2008為例)
STR((A,B))會被解讀為實參為:(A,B),而不是被解讀為兩個實參,第一個是(A第二個是B)。
(3) 有參宏定義中##的用法
#define WIDE(str) L##str
則會將形參str的前面加上L
比如:WIDE("abc")就會被替換成L"abc"
如果有#define FUN(a,b) vo##a##b()
那么FUN(id ma,in)會被替換成void main()
再比如:
#define s5(a) supper_ ## a
#include <stdio.h>
void supper_printf(const char* p )
{
printf("this is supper printf:\n%s\n",a);
}
int main()
{
s5(printf)("hello owrld");//就是調用函數supper_printf.
return 0;
}
(4) 多行宏定義:
#define doit(m,n) for(int i=0;i<(n);++i)\
{\
m+=i;\
}
關鍵是要在每一個換行的時候加上一個 "\ " ,最后一行不用加。這樣使用的時候就可以用doit(m,n)來代替for循環結構了。