最近在阅读redis源码,遇到这样的宏定义代码#define __str(s) #s
,在以前并没有看到过这样的宏定义方式,因此就查阅资料对此类特殊的宏定义进行一个总结。
’ # ’ 在宏定义中的使用
- 将参数设置转换为字符串,即给参数的加上 " " ;
#include<stdio.h>
#define __str(s) #s
int main(){
char *str = __str(hello world!);
printf("%s\n",str);
return 0;
}
上述代码将会打印出 hello world!
’ ## '在宏定义中的使用
’ ## '的作用便是将两个参数连接在一起,而这种连接就是直接将这两个参数拼接成一个整体,比如 整型12 和整型 14 拼接在一起则形成整型数据 1214,同样也可以拼接成double类型或者字符串类型。
#include<stdio.h>
#define __str(s) #s
#define __connect(a, b) a##b
int main() {
int a = __connect(12,14);
double b = __connect(14.1,23);
const char* str = __connect("hello", "world");
printf("%s\n%d\n%lf\n", str, a, b);
return 0;
}
上述代码将会打如下打印:
防止嵌套宏不会展开的情况
当我们在宏定义中使用了 # 或者 ## 的时候就会发现某些时候当我们嵌套的使用宏定义时,内层的宏定义并不会被展开。如下代码:
#include<stdio.h>
#define __str(s) #s
#define __connect(a, b) a##b
#define __sum(a, b) (a)+(b)
int main() {
const char* str = __str(__sum(1,2));
printf("%s\n",str);
return 0;
}
上述代码中我们将会的到这用的结果
如果我们想要将内层的宏定义__sum(1, 2)展开,那么我们定义两次宏,在redis源码中是这样的:
/* Double expansion needed for stringification of macro values. */
#define __xstr(s) __str(s)
#define __str(s) #s
整合到我上述的代码中:
#include<stdio.h>
#define __xstr(s) __str(s)
#define __str(s) #s
#define __connect(a, b) a##b
#define __sum(a, b) (a)+(b)
int main() {
int res = __sum(1, 2);
const char* str = __xstr(__sum(1,2));
printf("%s\n",str);
return 0;
}
此时的运行结果将会是我们想要的结果。