首先,必须说明的是,如果你用C++,应当尽可能的避免使用宏。
然后,宏有其强大和便捷之处。接下来,简单介绍一下,工作学习中,遇到的一些关于宏的使用技巧,算是对工作的一个总结。
1、#和##
#:把宏参数转换为字符串。不管该参数宏什么,即“原貌”用字符串显示出来
如#define Log(x) printf("%s\n",#x)
(1)、Log(10); //输出10,注意printf里面是“%s”而不是“%d”;
(2)、int nCount=10;
Log(nCount); // 输出的是nCount,而不是10
##被称为连接符(concatenation),把宏参数与之前的token(参数/字符串空格等)连接起来。
如
#define PARAM_INVILED (-1)
#define PARAM_INVILED_STR "参数错误"
#define ShowErrInfo(ERRNO) printf("[Code:%d]:%s\n",ERRNO,ERRNO##_STR);
输出:[Code:-1]:参数错误
2、宏嵌套宏
当有'#'或'##'的时候,宏嵌套宏并不能有效的展开,此时需要多加一层宏进行转换。
如:
#define INT_X 1
#define INT_Y 2
#define INT_SPLICE(x,y) (x##y)
printf("%d\n",INT_SPLICE(1,2));//没问题,输出12;
printf("%d\n",INT_SPLICE(INT_X,INT_Y));//编译报错;
此时,对宏定义需要如下处理:
#define _INT_SPLICE(x,y) (x##y)
#define INT_SPLICE(x,y) _INT_SPLICE(x,y)
则printf("%d\n",INT_SPLICE(INT_X,INT_Y));//输出12;
2、可变参数宏
在C宏中称为Variadic Macro,也就是变参宏。比如:
#define Log(fmt, ...) printf(fmt,##__VA_ARGS__)
#define Log(fmt,args...) printf(fmt,args)
第一个采用默认的宏__VA_ARGS__,##这个连接符号充当的作用就是当__VAR_ARGS__为空的时候,消除前面的那个逗号。
第二个宏中,我们显式地命名变参为args,在宏定义中就可以用args来代指变参.
可变参数宏其实比较简单,如下是我发现对printf的一些技巧,以前没有发现,可临时替代项目中的日志系统,输出到屏幕上,用于调试。
#ifdef WIN32
#define TrimFilePath(x) strrchr(x,'\\')?strrchr(x,'\\')+1:x
#else //*nix
#define TrimFilePath(x) strrchr(x,'/')?strrchr(x,'/')+1:x
#endif
#define LogDebug(fmt, ...) \
printf("[DEBUG] [%s(%d)] : " fmt"\n",TrimFilePath(__FILE__),__LINE__,##__VA_ARGS__)
#define LogInfo(fmt, ...) \
printf("[INFO ] [%s(%d)] : " fmt"\n",TrimFilePath(__FILE__),__LINE__,##__VA_ARGS__)
#define LogWarn(fmt, ...) \
printf("[WARN ] [%s(%d)] : " fmt"\n",TrimFilePath(__FILE__),__LINE__,##__VA_ARGS__)
#define LogError(fmt, ...) \
printf("[ERROR] [%s(%d)] : " fmt"\n",TrimFilePath(__FILE__),__LINE__,##__VA_ARGS__)
ZhaiPillary
2016-12-17