详谈#define的替换规则以及宏定义中的#和##详解

\underline{创作不易,感谢支持! }

创作不易,感谢支持!​

再看本篇文章时,建议先看一下#define的基本介绍

推荐文章:深入了解#define

链接: 深入了解预处理命令#define

回归正题

#define的替换规则总体分为以下几个步骤

1.在调用宏时,首先对参数进行检查,看看是否包含任何由#define定义的符号。如果是,它们会被替换。

什么意思呢?

比如说你定义的MAX为100

#define MAX 100

int main()

{

int ret = MAX;

return 0;

}

此时代码运行预编译(预处理)时,会把它替换成如下

#define MAX 100

int main()

{

int ret = 100;

return 0;

}

2.替换后被插入到程序中原来文本的位置,对于宏,参数名被他们替换。(意思是如果这时有两个参数)

示例:(这时你定义了两个参数,一个宏,一个标识符)

#define MAX 100

#define DOUBLE(X) ((X)+(X))

int main()

{

int ret = 10 * DOUBLE(MAX);

return 0;

}

此时代码运行预编译(预处理)时,会把它替换成如下

#define MAX 100

#define DOUBLE(X) ((X)+(X))

int main()

{

//int ret = 10 * DOUBLE(MAX);

int ret = 10 * ((100) + (100));

printf("%d\n", ret);

return 0;

}

注意:是先替换标识符MAX,然后替换宏DOUBLE

3.最后,再次扫描,看看是否包含任何由#define定义的符号。如果是,重复上述过程。

对执行的代码进行检查,如果还有遗漏,重复执行1 、2过程

最终替换成没有宏的现象!

注意:

1.宏参数和#define定义中可以出现其他#define定义的变量。但是对于宏,不能出现递归。(宏是没有递归的概念) 2.当预处理搜索#define定义的符号时,字符串常量的内容不被搜索

示例

注意:此时字符串中MAX是不需要替换的,因为这就是字符串本来具有的内容!

看到这里,相信大家也应该明白了#define的替换规则了!

接下来,继续说一下 # 和 ## 这两个操作符 注意:这两个符号是单独存在的,不是#define前面的#

单独的#是什么呢?

1.#

功能:它用于把参数插入到字符串中,只有宏定义中使用(#define) 总所周知,一个常量字符串中的内容是不能修改的

int main()

{

printf("hello boy");

return 0;

}

其中“”两个双引号之中的内容是不能修改的 但是有了 # 我们可以对常量字符串本身进行修改

示例:

在我们定义的宏里面如果出现#参数,会把X所对应的内容转换成对应的字符串

两个连在的#是什么呢?

2.##

功能:用于粘连两个标识符,只有宏定义中使用(#define)

用于把位于它两边的符号合成一个符号。它允许宏定义从分离的文本片段创建标识符

示例:

注意:这样连接必须产生一个合法的标识符(意思是必须提前声明过),否则其结果就是未定义的。

以上代码均在vs2022环境下编译