日志输出可变参数宏的使用记录
#define COLOR_RED "\033[1, 31m" #define COLOR_NONE "\033[0m"
#if DEBUG #define DBG_PRT(fmt, args...)\ do\ {\ printf(COLOR_RED"file:%s line:%s", __FILE__, __LINE__);\ printf(fmt,##args);\ }while(0)
#else
#define DGB_PRT(fmt, args...)
#endif
上述代码是在嵌入式日志打印中常用的宏函数,这个宏打印函数可以显示打印信息所在的文件、行数、函数名等一些你需要的信息。另外需要注意的是,在宏定义中尽量使用do while(0)结构来包含内容,如果用花括号{}来包含往往会出现一些错误,比如我将上述代码改成小面用花括号包含的形式:
#define DBG_PRT(fmt, args...)\ {\ printf(COLOR_RED"file:%s line:%s", __FILE__, __LINE__);\ printf(fmt, ##args);\ } if (1) DGB_PRT("hello world"); 展开后为: if (1) { printf(COLOR_RED"file:%s line:%s", __FILE__, __LINE__);printf(args); };-->这里会有一个;号,会有语法错误
上述DGB_PRT(fmt, args...)定义中, “...”符号表示的是可变参数,args...的写法表示在后续的使用中
可以用args标识符表示可变参数列表,然后在后面的宏定义中,直接使用 args 代表变参列表就可以了。
我们在标识符args 前面加上宏连接符 ##,这样做的好处是,当变参列表非空时,## 的作用是连接 fmt,和变参列表,
各个参数之间用逗号隔开,宏可以正常使用;当变参列表为空时,## 还有一个特殊的用处,它会将固定参数 fmt 后面的逗号删除掉,
这样宏也就可以正常使用了。
可变参数宏的另外两种写法
第一种,使用__VA_ARGS__表示可变参数列表
#define DBG_PRT(fmt, ...)\ do\
{\ printf(COLOR_RED"file:%s line:%s", __FILE__, __LINE__);\ printf(fmt, ##__VA_ARGS__);\ }while(0)
第二种写法比较简洁,直接将可变参数列表...和fmt标识符连在一起,这样后续使用的时候直接使用fmt标识符,这个标识符就同时包含了格式化字符串和可变参数列表,同时可以应对参数为空的情形,因为可变参数列表和fmt是连接在一起的,可变参数列表为空时,最后fmt表示的内容就只有格式化字符串了。
#define DBG_PRT(fmt...)\ do\ {\ printf(COLOR_RED"file:%s line:%s", __FILE__, __LINE__);\ printf(fmt);\ }while(0)
另外记录一下#和##的使用方法(参考https://blog.csdn.net/baidu_33850454/article/details/79363033)
使用#把宏参数变为一个字符串——字符串化操作符
用##把两个宏参数贴合在一起——字符连接操作符
一般用法
#include<cstdio> #include<climits> using namespace std; #define STR(s) #s #define CONS(a,b) int(a##e##b) int main() { printf(STR(vck)); // 输出字符串"vck" printf("%d\n", CONS(2,3)); // 2e3 输出:2000 return 0; }
注意: 当宏参数是另一个宏的时候,需要注意的是凡宏定义里有用’#’或’##’的地方宏参数是不会再展开.
即, 只有当前宏生效, 参数里的宏!不!会!生!效 !!!!
#define A (2) #define STR(s) #s #define CONS(a,b) int(a##e##b) printf("int max: %s\n", STR(INT_MAX)); // INT_MAX # printf("%s\n", CONS(A, A)); // compile error --- int(AeA) 展开后: printf("int max: %s\n","INT_MAX"); printf("%s\n", int(AeA));
由于A和INT_MAX均是宏,且作为宏CONS和STR的参数,并且宏CONS和STR中均含有#或者##符号,所以A和INT_MAX均不能被解引用。导致不符合预期的情况出现。
解决这个问题的方法很简单. 加多一层中间转换宏. 加这层宏的用意是把所有宏的参数在这层里全部展开,那么在转换宏里的那一个宏(_STR)就能得到正确的宏参数.
#define A (2)
#define _STR(s) #s
#define STR(s) _STR(s) // 转换宏
#define _CONS(a,b) int(a##e##b)
#define CONS(a,b) _CONS(a,b) // 转换宏
printf("int max: %s\n",STR(INT_MAX));
//输出为: int max:0x7fffffff
//STR(INT_MAX) --> _STR(0x7fffffff)
然后再转换成字符串; printf("%d\n", CONS(A, A));
//输出为:200
//CONS(A, A) --> _CONS((2), (2)) --> int((2)e(2))