我的日志库中有以下宏:
#define TRACE_E(__logCat,__format,...) \
do { \
::dbg::LogCategory * const __catPtrVal = (::dbg::LogCategory *)(__logCat); \
if( NULL != __catPtrVal && __catPtrVal->IsEnabled() ) \
{ \
__catPtrVal->Error( __format, __VA_ARGS__ ); \
} \
} while( false )
在 Visual Studio (2008) 下,它按预期工作,即我可以同时执行 TRACE_E( pLog, "some message without parameters");
和 TRACE_E( pLog, "some message with参数 %d %d", 4, 8 );
但是,当我将同一个库与 eclipse 和 Android NDK 一起使用时,如果我没有在我的宏中的格式字符串之后传递至少一个参数,即 TRACE_E( pLog, "some message without parameters");
无效,但是 TRACE_E( pLog, "some message without parameters", 0 );
是,这迫使我在不需要时传递一个虚拟参数.
当使用 g++ 而不是 Visual Studio 的编译器时,可变参数宏的行为有什么不同吗?谢谢。
最佳答案
是的。您所尝试的在标准 C 或 C++ 中是不可能的。
这可以说是各个标准中的一个缺陷,不同的编译器有不同的解决方法。 Visual Studio 试图让它按原样工作,gcc 和 clang 需要以下语法:
__catPtrVal->Error( __format, ##__VA_ARGS__ );
这描述了here对于海湾合作委员会; clang 只是采用了 gcc 的做事方式。不幸的是,MSVC 不理解这种语法。据我所知,在一般情况下没有可移植的方法来解决这个问题。
不过,对于您的特定宏,您可以简单地编写
#define TRACE_E(__logCat,...) \
do { \
::dbg::LogCategory * const __catPtrVal = (::dbg::LogCategory *)(__logCat); \
if( NULL != __catPtrVal && __catPtrVal->IsEnabled() ) \
{ \
__catPtrVal->Error(__VA_ARGS__ ); \
} \
} while( false )
因为您唯一使用 __format
的地方就在 __VA_ARGS__
之前。
旁注:您使用了很多 reserved identifiers那里。除非您正在编写标准库实现,否则您应该更轻松地使用下划线。
关于Android - NDK - 需要 1+ args 的 Variadic 宏,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29557649/