printf() 和 fprintf() 这些输出函数的参数是可变的,在调试程序时,你可能希望定义自己的参数可变的输出函数,那么可变参数宏会是一个选择。
C99 中规定宏可以像函数一样带有可变参数,比如

  1. #define LOG(format, ...) fprintf(stdout, format, \_\_VA\_ARGS\_\_)

其中,… 表示参数可变,VA_ARGS在预处理中为实际的参数集所替换
GCC 中同时支持如下的形式

  1. #define LOG(format, args...) fprintf(stdout, format, args)

其用法和上面的基本一致,只是参数符号有变化

有一点需要注意,上述的宏定义不能省略可变参数,尽管你可以传递一个空参数,这里有必要提到 “##” 连接符号的用法。

“##” 的作用是对 token 进行连接,在上例中,format、VA_ARGS、args 即是 token,
如果 token 为空,那么不进行连接,所以允许省略可变参数 (VA_ARGS和 args),对上述变参宏做如下修改

  1. #define LOG(format, ...) fprintf(stdout, format, ##\_\_VA\_ARGS\_\_)
  1. #define LOG(format, args...) fprintf(stdout, format, ##args)

上述的变参宏定义不仅能自定义输出格式,而且配合 #ifdef #else #endif 在输出管理上也很方便,

比如调试时输出调试信息,正式发布时则不输出,可以这样

  1. #ifdef DEBUG
  2. #define LOG(format, ...) fprintf(stdout, ">>"format"\n", ##__VA_ARGS__)
  3. #else
  4. #define LOG(format, ...)

在调试环境下,LOG 宏是一个变参输出宏,以自定义的格式输出;在发布环境下,LOG 宏是一个空宏,不做任何事情。

参考资料