define宏定义中的#,##,@#及符号
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
d efine宏定义中的#,##,@#及\符号(ZT)
C++ STL学习2011-04-24 18:04:03 阅读19 评论0 字号:大中小订阅
1、# (stringizing)字符串化操作符。其作用是:将宏定义中的传入参数名转换成用一对双引号括起来参数名字符串。其只能用于有传入参数的宏定义中,且必须置于宏定义体中的参数名前。
如:
#define example(instr) printf("the input string is:\t%s\n",#instr)
#define example1(instr) #instr
当使用该宏定义时:
example(abc);在编译时将会展开成:printf("the input string is:\t%s\n","abc");
string str=example1(abc);将会展成:string str="abc";
注意:
对空格的处理
a。忽略传入参数名前面和后面的空格。
如:str=example1( abc );将会被扩展成str="abc";
b.当传入参数名间存在空格时,编译器将会自动连接各个子字符串,用每个子字符串中只以一个空格连接,忽略其中多余一个的空格。
如:str=exapme( abc def); 将会被扩展成str="abc def";
2、## (token-pasting)符号连接操作符
宏定义中:参数名,即为形参,如#define sum(a,b) (a+b);中a和b 均为某一参数的代表符号,即形式参数。
而##的作用则是将宏定义的多个形参成一个实际参数名。
如:
#define exampleNum(n) num##n
int num9=9;
使用:
int num=exampleNum(9); 将会扩展成int num=num9;
注意:
1.当用##连接形参时,##前后的空格可有可无。
如:#define exampleNum(n) num ## n 相当于#define exampleNum(n) num##n
2.连接后的实际参数名,必须为实际存在的参数名或是编译器已知的宏定义
// preprocessor_token_pasting.cpp
#include
#define paster( n ) printf_s( "token" #n " = %d", token##n )
int token9 = 9;
int main()
{
paster(9);
}
运行结果:
token9 = 9
3、@# (charizing)字符化操作符。
只能用于有传入参数的宏定义中,且必须置于宏定义体中的参数名前。作用是将传的单字符参数名转换成字符,以一对单引用括起来。#define makechar(x) #@x
a = makechar(b);
展开后变成了:
a= 'b';
4、\ 行继续操作符
当定义的宏不能用一行表达完整时,可以用"\"表示下一行继续此宏的定义。
注意:换行不能切断单词,只能在空格的地方进行。
在#define中,标准只定义了#和##两种操作。#用来把参数转换成字符串,##则用来连接两个前后两个参数,把它们变成一个字符串。
#include
#define paster( n ) printf( "token " #n" = %d\n ", token##n )
int main()
{
int token9=10;
paster(9);
return 0;
}
输出为
[leshy@leshy src]$ ./a.out
token 9 = 10
宏定义的特殊符号# ## - [C++]
1. 利用宏参数创建字符串:# 运算符
在类函数宏(function-like macro)的替换部分中,“#”符号用作一个预处理运算符,它可以把语言符号(token)转化为字符串。例如,如果x 是一个宏参量,那么#x 可以把参数名转化为相应的字符串。该过程称为字符串化。
说明:类函数宏就是带参数的宏。类函数宏的定义中,用圆括号括起来一个或多个参数,随后这些参数出现在替换部分。
#include
#define PSQR(x) printf("The square of " #x " is %d. \r\n", (x) * (x))
int main(void)
{
int y = 5;
PSQR(y);
PSQR(2 + 4);
return 0;
}
// 输出:
The square of y is 25. // 用"y" 代替#x
The square of 2 + 4 is 36. // 用"2 + 4" 代替#x
#include
#define PSQR(x) printf("The square of " #x " is %d. \r\n", (x) * (x))
int main(void)
{
int y = 5;
PSQR(y);
PSQR(2 + 4);
return 0;
}
// 输出:
The square of y is 25. // 用"y" 代替#x
The square of 2 + 4 is 36. // 用"2 + 4" 代替#x
#define STRING2(x) #x
#define STRING(x) STRING2(x)
#define WQ wangqi
#pragma message(STRING2(WQ)) // WQ(字符串)
#pragma message(STRING(WQ)) // wangqi(字符串)
#define STRING2(x) #x
#define STRING(x) STRING2(x)
#define WQ wangqi
#pragma message(STRING2(WQ)) // WQ(字符串)
#pragma message(STRING(WQ)) // wangqi(字符串)
2. 预处理器的粘合剂:## 运算符
和# 运算符一样,## 运算符可以用于类函数宏的替换部分。另外,## 运算符还可用于类对象宏(object-like macro)的替换部分。这个运算符把两个语言符号组合成单个语言符号。例如,可以定义如下宏:
#define XNAME(n) x ## n
#define XNAME(n) x ## n
宏调用XNAME(4) 会展开成x4 。
说明:类对象宏就是用来代表值的宏。如,#define PI 3.141593 中的PI。
#include
#define XNAME(n) x ## n
#define PRINT_XN(n) printf("x" #n " = %d\r\n", x ## n);
int main(void)
{
int XNAME(1) = 14; // 变为int x1 = 14;
int XNAME(2) = 20; // 变为int x2 = 20;
PRINT_XN(1) // 变为printf("x1 = %d\r\n", x1);
PRINT_XN(2) // 变为printf("x2 = %d\r\n", x2);