c宏定义#define、#、##、​​__VA_ARGS__​​符号的理解


​#define​​ 宏定义就是预处理时字符串替换的过程,不涉及内存分配,效率比较高,所以简单的表达式有些人就喜欢使用宏定义实现
注意:宏定义只是替换操作,不进行任何逻辑运算操作,所以不涉及内存分配


//例子1
#define M 5 //将M替换为5
printf("结果 = %d\n", M) //编译器预处理后实际为 printf("结果 = %d\n", 5)

//例子2
#define SUM(a,b) a+b //SUM和(a,b)之间不能用空格,不然就变为将SUM替换为(a,b) a+b了
printf("结果 = %d\n", SUM(1,2)) //编译器预处理后实际为 printf("结果 = %d\n", 1+2)

//例子3,有参宏定义时最好加上括号,以免替换后意思变了
#define COUNT(M) M * M
printf("结果 = %d\n", COUNT(1+2)) //实际为 printf("结果 = %d\n", 1+2 * 1+2) 结果是5,而不是想要的9。。。

//例4
#define COUNT(M) ((M) * (M))
printf("结果 = %d\n", COUNT(1+2)) //实际为 printf("结果 = %d\n", ((1+2) * (1+2)) ) 结果是9,是对的

#define COUNT(M) ((M) * (M))
//使用COUNT宏定义
COUNT(10) 相当于
#define SUM(a,b) a + b


​#运算符​​ 把参数字符串化,功能就是转为字符串


//例子
#define SUM(a, b) printf(#a " + " #b " = %d\n",((a) + (b)))
调用 SUM(1 + 2, 3 + 4);
相当于 printf("1 + 2" " + " "3 + 4" " = %d\n",((1 + 2) + (3 + 4))) //#a替换为"1 + 2",#b替换为"3 + 4"
字符串本身具有链接功能,所以相当于 printf("1 + 2 + 3 + 4 = %d\n",(3 + 7))


​##运算符​​ 参数拼接粘合作用,所以称为“预处理器的粘合剂”


//例子
#define NAME(n) num ## n
int num0 = 100; //必须定义num0变量,不然下面的NAME(0)调用时会提示找不到num0变量
printf("num0 = %d\n", NAME(0)); //实际为 printf("num0 = %d\n", num0); 也就是NAME(0)替换为了num0


​__VA_ARGS__​​符号 可变参数列表,如果是多个参数,必须是最后一个参数,代表剩余的。


//例子__VA_ARGS__表示前面的...所有参数,前面再加个#表示字符串化即转为字符串形式
#define LOG(...) printf("log-->%s\n",#__VA_ARGS__)
LOG("hello,world"); //打印出 log-->"hello,world"
LOG(123); //打印出 log-->123

//例子2
#define LOG(...) printf("log-->%s\n",#__VA_ARGS__)
调用:LOG("hello,world");
调用:LOG(123);

#define LOG2(TAG, ...) printf(TAG"-->%s\n",#__VA_ARGS__)
调用:LOG2("TAG", "hello,world"); //必须带引号"TAG",不能是TAG

#define LOG3(TAG, ...) printf(#TAG"-->%s\n",#__VA_ARGS__)
LOG3(TAG, 123); //不能带引号"TAG",只能是TAG当然也可以是其他的但是不能带引号,因为#define时会添加了#转为字符串

小技巧:使用CLION或者开发工具,调用宏定义时,工具会出现Replacement的替换提示结果,我们可以工具提示查看替换后的结果是不是我们想要的