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。除非你正在编写标准库实现,否则你应该在下划线上更容易。