C/C++ 的宏中,#的功能是将其后面的宏参数进行字符串化操作,简单说就是在对它所引用的宏变量通过替换后在其左右各加上一个双引号。
##连接符号由两个井号组成,其功能是在带参数的宏定义中将两个子串联接起来,从而形成一个新的子串。但它不可以是第一个或者最后一个子串。
凡是宏定义里有用#或##的地方宏参数是不会再展开,例如_STRI(INT_MAX)中的INT_MAX就不会被展开为2147483647。
如果想要使其中的宏参数展开,则需要多加一层中间转换宏: #define STRI(s) _STRI(s)
加这层宏的用意是把所有宏的参数在这层里全部展开,那么在转换宏里的宏就能得到对应的宏参数。
#include<stdio.h>
#include<iostream>
using namespace std;
#define f(a,b) a##b
#define g(a) #a
#define h(a) g(a)
#define A(x) T_##x
#define B(x) #@x
#define C(x) #x
#define WARN_IF(EXP) if(EXP) cerr << #EXP << endl;
#define paster( n ) cout << "token" << #n << " = " << n << endl;
#define _CONS(a, b) int(a##+##b)
#define _STRI(s) #s
#define STRI(s) _STRI(s)
void test_sharp_symbol()
{
printf("%s\n", h(f(1, 2))); //输出:12
printf("%s\n", g(f(1, 2))); //输出:f(1,2)
printf("%s\n", h(A(1))); // A(1)------〉T_1
printf("%d\n", B(1)); // B(1)------〉'1'
printf("%s\n", C(1)); // C(1)------〉"1"
int div = 0;
WARN_IF(div == 0); //输出: div == 0
paster(9); //输出: token9 = 9
cout << _CONS(1+2, 2) << endl; //输出: 3
cout << _STRI(INT_MAX) << endl; //输出: INT_MAX
cout << STRI(INT_MAX) << endl; // prints : 2147483647
}