C语言 ' # ' 在宏定义中的使用

199 阅读3分钟

@TOC 本文已参与「新人创作礼」活动,一起开启掘金创作之路。 最近发现学习 C++ 遇到了一些瓶颈,所以打算有空就积累几道面试题,复习巩固一下之前学习的知识。

' # ' 在宏定义中的使用

  1. 将参数设置转换为字符串,即给参数的加上 " " ;
#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;
}

此时的运行结果将会是我们想要的结果。 在这里插入图片描述


最近在阅读redis源码,遇到这样的宏定义代码#define __str(s) #s,在以前并没有看到过这样的宏定义方式,因此就查阅资料对此类特殊的宏定义进行一个总结。

' # ' 在宏定义中的使用

  1. 将参数设置转换为字符串,即给参数的加上 " " ;
#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;
}

此时的运行结果将会是我们想要的结果。 在这里插入图片描述