一种流行的技巧是用一个单独的用括弧括起来的的“参数” 定义和调用宏,
参数在宏扩展的时候成为类似printf() 那样的函数的整个参数列表。
#define DEBUG(args) (printf("DEBUG: "), printf args)
if(n != 0) DEBUG(("n is %d\n", n));
明显的缺陷是调用者必须记住使用一对额外的括弧。
gcc 有一个扩展可以让函数式的宏接受可变个数的参数。
但这不是标准。
另一种可能的解决方案是根据参数个数使用多个宏(DEBUG1, DEBUG2, 等等),
或者用逗号玩个这样的花招:
#define DEBUG(args) (printf("DEBUG: "), printf(args))
#define _ ,
DEBUG("i = %d" _ i);
C99 引入了对参数个数可变的函数式宏的正式支持。
在宏“原型” 的末尾加上符号. . . (就像在参数可变的函数定义中),
宏定义中的伪宏VA ARGS 就会在调用是替换成可变参数。
最后, 你总是可以使用真实的函数, 接受明确定义的可变参数。
参见问题15.4和15.5。
如果你需要替换宏, 使用一个函数和一个非函数式宏,
如#define printf myprintf。