0

我有以下两个宏:

#define F1(...) [NSString stringWithFormat:__VA_ARGS__]
#define F2(format, ...) [NSString stringWithFormat:(format), ##__VA_ARGS__]

当我嵌套它们时,F1 可以工作,但 F2 无法编译。

这段代码:

    F1(@"%@", F1(@"%@", @"a"));
    F2(@"%@", F2(@"%@", @"a"));

导致以下 clang 错误:

foo.m:51:15: warning: implicit declaration of function 'F2' is invalid in C99 [-Wimplicit-function-declaration]
    F2(@"%@", F2(@"%@", @"a"));
          ^
foo.m:18:64: note: expanded from macro 'F2'
#define F2(format, ...) [NSString stringWithFormat:(format), ##__VA_ARGS__]
                                                           ^
foo.m:51:15: warning: format specifies type 'id' but the argument has type 'int' [-Wformat]
    F2(@"%@", F2(@"%@", @"a"));
         ~~   ^~~~~~~~~~~~~~~
         %d
foo.m:18:64: note: expanded from macro 'F2'
#define F2(format, ...) [NSString stringWithFormat:(format), ##__VA_ARGS__]

请注意,F1 没有警告或错误,只有 F2 有问题。

这里发生了什么:这是预期的行为还是clang中的错误?

更重要的是,有没有办法让我调整 F2 以使其工作。我真的更喜欢 F2 的语法。


更新

遵循@KenThomases 的建议,我跑了clang -E -o foo.i foo.m

# 1 "foo.m"
# 1 "<built-in>" 1
# 1 "<built-in>" 3
# 178 "<built-in>" 3
# 1 "<command line>" 1
# 1 "<built-in>" 2
# 1 "foo.m" 2



NSLog([NSString stringWithFormat:@"%@", [NSString stringWithFormat:@"%@", @"a"]]);
NSLog([NSString stringWithFormat:(@"%@"), F2(@"%@", @"a")]);

我将此作为缺陷提交给 llvm.org:错误 19141

4

1 回答 1

1

在我看来,预处理器并没有扩展F2宏的内部使用。显然,使用##__VA_ARGS__只是F2逐字插入外部使用的参数,而不是进行进一步的扩展。

你可以让编译器预处理代码,它会告诉你发生了什么。 cc -E -o foo.i foo.c

于 2014-03-14T04:26:05.117 回答