背景:我们一般在调试加log时可能会看到这些东西,一般我是直接在代码中找别人能用的打印log的接口,现对该部分内容进行一下了解:
1,#:字符串化操作。将传入的宏参数名转换成字符串;
例:
#include<stdio.h>
#define PSQR(x) printf("the square #x(%s) (x)*(x)=%d.\n",#x,(x)*(x))
int main(void)
{
int y =4;
PSQR(y); //这里注意区分:宏参数名是 y
PSQR(1+4); //这里注意区分:宏参数名是 1+4
return 0;
}
输出结果:
the square #x(y) (x)*(x)=16.
the square #x(1+4) (x)*(x)=25.
结:第一次调用宏时使用“y”代替#x;第二次调用时用“1+4"代#x。
2,##:参数连接操作。将传入的宏参数名和其他部分粘合起来,组成一个字符串;
例:
#define NAME(n) x##n
这样宏调用:
NAME(4)
展开后:
x4
程序:
#include <stdio.h>
#define NAME(n) x##n
#define PXN(n) printf("x"#n" = %d\n",x##n)
int main(void)
{
int NAME(1)=2; //相当于int x1=2; 这里注意区分:宏参数名是 1
PXN(1); //相当于printf("x1 = %d\n", x1); 这里注意区分:宏参数名是 1
int m =3;
int NAME(m)=4; //相当于int xm=4; 这里注意区分:宏参数名是 m
PXN(m); //相当于printf("xm = %d\n", xm); 这里注意区分:宏参数名是 m
return 0;
}
输出结果:
x1 = 2
xm = 4
3,可变参数宏
(1)形式: … 与__VA_ARGS__ ;
实现思想就是宏定义中参数列表的最后一个参数为省略号(也就是三个点)。这样预定义宏__VA_ARGS__就可以被用在替换部分中,替换省略号所代表的字符串。比如:
例:
#include<stdio.h>
#define PR(...) printf(__VA_ARGS__) //
int main(void)
{
int w=1,h=2;
PR("hello\n");
PR("w = %d, h = %d\n",w,h);
return 0;
}
输出结果:
hello
w = 1, h = 2
注意:
如果有多个参数,省略号只能在参数列表的最后一个参数的位置。
如:#define W(…,x,y) 是错误的 !
#define W(x,…,y) 是错误的 !
#define W(x,y,…) 是正确的 !
(2)形式: args… 与 args ;
结:形式 “args… 与args” 和 形式 “… 与__VA_ARGS__” 是等价的;
例:
include<stdio.h>
#define PR(arg...) printf(arg) //注意:这里不是非要写成args才行
int main(void)
{
int w=1,h=2;
PR("hello\n");
PR("w = %d, h = %d\n",w,h);
return 0;
}
输出结果:
hello
w = 1, h = 2
注意:
在__VA_ARGS__ 或者 args 前面加 ## 的作用:当可变参数的个数为0时,这里的 ## 起到把前面多余的","去掉的作用,否则会编译出错;
#define LOG(format, ...) fprintf(stdout, format, ##__VA_ARGS__)
#define LOG(format, args...) fprintf(stdout, format, ##args)