Android - NDK - 可变参数宏需要 1+ 个参数
Android - NDK - Variadic macro requiring 1+ args
我的日志库中有以下宏:
#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 parameters %d %d", 4, 8 );
但是当我将同一个库与 eclipse 和 Android NDK 一起使用时,如果我没有在宏中的格式字符串后传递至少一个参数,即 TRACE_E( pLog, "some message without parameters" );
无效,但 TRACE_E( pLog, "some message without parameters", 0 );
有效,这迫使我在需要 none 时传递一个虚拟参数。
使用 g++ 而不是 Visual Studio 的编译器时,可变参数宏的行为有什么不同吗?谢谢。
是的。您尝试的操作在标准 C 或 C++ 中是不可能的。
这可以说是各个标准中的一个缺陷,不同的编译器有不同的解决方法。 Visual Studio 试图让它按原样工作,gcc 和 clang 需要以下语法:
__catPtrVal->Error( __format, ##__VA_ARGS__ );
这是针对 gcc 的描述 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。除非你正在编写标准库实现,否则你应该在下划线上更容易。
我的日志库中有以下宏:
#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 parameters %d %d", 4, 8 );
但是当我将同一个库与 eclipse 和 Android NDK 一起使用时,如果我没有在宏中的格式字符串后传递至少一个参数,即 TRACE_E( pLog, "some message without parameters" );
无效,但 TRACE_E( pLog, "some message without parameters", 0 );
有效,这迫使我在需要 none 时传递一个虚拟参数。
使用 g++ 而不是 Visual Studio 的编译器时,可变参数宏的行为有什么不同吗?谢谢。
是的。您尝试的操作在标准 C 或 C++ 中是不可能的。
这可以说是各个标准中的一个缺陷,不同的编译器有不同的解决方法。 Visual Studio 试图让它按原样工作,gcc 和 clang 需要以下语法:
__catPtrVal->Error( __format, ##__VA_ARGS__ );
这是针对 gcc 的描述 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。除非你正在编写标准库实现,否则你应该在下划线上更容易。