宏的使用的核心,就是替换,换是最关键的。
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循环结构了。