1

我有一项工作要使静态库更安全,为此我需要替换 的格式化字符串以printf使它们在编译的静态库中以不同的方式出现,为此必须在预处理器中完成阶段 。

我所做的(它实际上在 Visual Studio 中有效)如下(这只是一个伪示例):

char * my_array[] = {"abcd", "a %d", " b %d %s "};
#define GENERIC_ARRAY(x) my_array[x]

#define VARIADIC_DEBUG_PRINT(...)   DebugPrintFunction (__FILE__, __LINE__, __func__, __VA_ARGS__)
#define PRINT_BY_LEVEL(x)           VARIADIC_DEBUG_PRINT x
#define REPLACE_STRING(x,...)       PRINT_BY_LEVEL((GENERAL_LEVEL,GENERIC_ARRAY(__COUNTER__),__VA_ARGS__))

#define MY_PRINTF(x,...)      REPLACE_STRING((void*)0,(void*)0,__VA_ARGS__)

所有这些开销都是为了欺骗编译器接受除了字符串之外没有任何参数的打印

因此,在我的 main.c 中对其进行测试时,我尝试了以下方法并且它起作用了:

MY_PRINTF("Hello World");
MY_PRINTF("My Val %d", i);
MY_PRINTF("MY VAL %d My String %s", i, s);

但是当切换到 GCC 时,他不喜欢第一个打印的格式,即:

MY_PRINTF("Hello World");

并给我一个编译错误:

error: expected expression before ')' token

有什么想法可以欺骗编译器并接受它吗?或者更好的想法如何在编译后安全地重命名字符串?

4

2 回答 2

4

您可以尝试以下方法:

#include <stdio.h>
#define PRINT(x, ...) printf(x, ##__VA_ARGS__)

int main (int argc, char *argv[]) {
    PRINT("Hello\n");
    PRINT("World %d\n", 42);
    return 0;
}

它适用于 gcc 4.8(未尝试使用早期版本,但它也应该适用)

于 2014-01-20T13:50:34.200 回答
1

使用##__VA_ARGS__,您可以尝试:

#define MY_PRINTF(x, ...) \ 
    VARIADIC_DEBUG_PRINT(GENERAL_LEVEL, GENERIC_ARRAY(__COUNTER__), (void*)0, (void*)0, ##__VA_ARGS__)
于 2014-01-20T14:38:24.927 回答