日志输出可变参数宏的使用记录

#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))

 

posted @ 2020-10-28 14:32  SkrSky  阅读(789)  评论(0编辑  收藏  举报