粘贴形成了一个无效的处理标记“。”

Pasting formed an invalid processing token '.'

我正在尝试使用宏根据类型调用适当的对象。

#define DELEGATE_FUNC(FuncName, kind, paramPtr)         \
    if (kind == 1) {                                    \
        return PolicyObject1.##FuncName(paramPtr);      \
    }                                                   \
    else {                                              \
        return PolicyObject2.##FuncName(paramPtr);      \
    }                                                   \
    return 0;                                           \

(PolicyObject1 & PolicyObject2 是两个静态对象。) 现在使用宏时,例如

DELEGATE_FUNC(ProcessPreCreate, 1, null_ptr);

它在 VS 2015 中编译良好,但在 LLVM 中出错 "Pasting formed an invalid processing token '.ProcessPreCreate'"

我查找并找到了一些帖子并在一定程度上理解了它 - 需要双重间接级别,例如 Why do I need double layer of indirection for macros?

但是我无法定义那两层宏,谁能帮忙?

(设计方面的讨论请先搁置)

谢谢

当编译器读取您的 C++ 文件时,第一步是将其分成标记,例如标识符、字符串文字、数字、标点符号等。C 预处理器处理这些 标记,而不是 text## 运算符将标记粘合在一起。因此,例如,如果您有

#define triple(foo) foo##3

然后 triple(x) 将为您提供标识符 x3triple(12) 将为您提供整数 123triple(.) 将为您提供浮点数 .3.

但是,您拥有的是 .##FuncName,其中 FuncNameProcessPreCreate。这将创建单个标记 .ProcessPreCreate,它不是有效的 C++ 标记。如果您直接键入 PolicyObject1.ProcessPreCreate 而不是通过宏,它将被标记为三个标记:PolicyObject1.ProcessPreCreate。这是您的宏需要生成的内容,以便提供有效的 C++ 输出。

要做到这一点,只需去掉 ##。没有必要将 . 粘附到 FuncName,因为它们是单独的标记。要检查这一点,您可以在 . 和成员名称之间放置一个 space;它仍然会编译得很好。由于它们是单独的令牌,因此不应也不能将它们粘合在一起。

删除“##”。

#define DELEGATE_FUNC(FuncName, kind, paramPtr)     \
if (kind == 1) {                                    \
    return PolicyObject1.FuncName(paramPtr);        \
}                                                   \
else {                                              \
    return PolicyObject2.FuncName(paramPtr);        \
}                                                   \
return 0;                                           \