#include "stdafx.h"
#include <string>
using namespace std;
#include <iostream>
void test()
{
{
/*
# (stringizing)字符串化操作符。
其作用是:将宏定义中的传入参数名转换成用一对双引号括起来参数名字符串。
其只能用于有传入参数的宏定义中,且必须置于宏定义体中的参数名前。
*/
#define example(instr) printf("the input string is:%s\n", #instr)
#define example1(instr) #instr
//忽略传入参数名前面和后面的空格
example(abc);//宏定义展开为 printf("the input string is:%s\n", #abc)
string str = example1(hello);
cout << str << endl;
//当传入参数名间存在空格时,编译器将会自动连接各个子字符串,用每个子字符串中只以
//一个空格连接,忽略其中多余一个的空格
example(abc def);
}
{
/*
## (token-pasting)符号连接操作符
宏定义中:参数名,即为形参,如#define sum(a,b) (a+b);
中a和b均为某一参数的代表符号,即形式参数。
而##的作用则是将宏定义的多个形参成一个实际参数名
注意:
1.当用##连接形参时,##前后的空格可有可无。
如:#define exampleNum(n) num ## n 相当于 #define exampleNum(n) num##n
2.连接后的实际参数名,必须为实际存在的参数名或是编译器已知的宏定义
另外,如果##后的参数本身也是一个宏的话,##会阻止这个宏的展开。
*/
#define paster(n) printf_s("token" #n "=%d\n", token##n)
#define STRCPY(a, b) strcpy(a ## _p, #b)
int token9 = 10;
paster(9);
}
{
char var1_p[20];
char var2_p[30];
strcpy(var1_p, "aaaa");
strcpy(var2_p, "bbbb");
STRCPY(var1, var2);
STRCPY(var2, var1);
printf("var1 = %s\n", var1_p);// ## 阻止了参数的宏展开!
printf("var2 = %s\n", var2_p);// 如果宏定义里没有用到 # 和 ##, 宏将会完全展开
}
{
#define A1(name, type) type name_##type##_type
#define A2(name, type) type name##_##type##_type
A1(a1, int); /* 等价于: int name_int_type; */
A2(a1, int); /* 等价于: int a1_int_type; */
}
{
//单独一个#,则表示 对这个变量替换后,再加双引号引起来
#define __stringify_1(x) #x
//那么 __stringify_1(linux) <= = > "linux"
printf("str = %s\n", __stringify_1(hello world));
}
{
#define CCC DDD
#define BBB CCC
#define AAA BBB
#define __stringify_1(x) #x
#define __stringify(x) __stringify_1(x)
#define __stringify_src(x) #x
//Doing two levels allows the parameter to be a macro itself.
printf("stringify(AAA): %s\n", __stringify(AAA));
printf("stringify_src(AAA): %s\n", __stringify_src(AAA));
}
}
int _tmain(int argc, _TCHAR* argv[])
{
test();
system("pause");
return 0;
}