为什么此宏没有扩展?

时间:2020-04-02 20:48:08

标签: c++ c-preprocessor

这是assets.cpp.i的预处理器输出:

#define CLASSNAME GAMEOBJECT
#define CLASSNAME_NORMAL GameObject
#define GAMEOBJECT_EXPANSION(FUNC) FUNC(std::vector<std::shared_ptr<Component>>, comps)




# 1 "/home/rubend/p/engine/libs/include/prefab.h" 1

#define prefab_h 



#define PREFAB_VAR_ENUM_NAME(name) name ##_enum
#define PREFAB_VAR_ENUM_DIRTY(type,name) PREFAB_VAR_ENUM_NAME(name),


# 42 "/home/rubend/p/engine/libs/include/prefab.h"
#define _CLASSNAME_EXPANSION(classname,func) classname ## _EXPANSION(func)
#define CLASSNAME_EXPANSION(func) _CLASSNAME_EXPANSION(CLASSNAME,func)

template<class T>
class Prefab:public File{
private:
 std::shared_ptr<T> ref;

protected:

 enum DIRTY_ENUM{First, CLASSNAME_EXPANSION(PREFAB_VAR_ENUM_DIRTY) Last};

};

现在CLASSNAME_EXPANSION(PREFAB_VAR_ENUM_DIRTY)为什么根本不扩展?

我不知道的有关连接令牌##的特别之处吗?

我正在使用GCC 9.3和CMake 3.16.5

答案

通过添加另一个宏将其修复:

#define __CLASSNAME_EXPANSION(classname, func) classname ## _EXPANSION(func)
#define _CLASSNAME_EXPANSION(classname, func) __CLASSNAME_EXPANSION(classname,func)
#define CLASSNAME_EXPANSION(func) _CLASSNAME_EXPANSION(CLASSNAME,func)

更新 12/4/20

@rici指出,我应该在下划线后面加上下划线,因为内部库宏在开头使用了下划线。

1 个答案:

答案 0 :(得分:3)

如果您进行扩展,则会注意到CLASSNAME_EXPANSION正在扩展-本身。

让我们逐步了解发生的情况:

CLASSNAME_EXPANSION(PREFAB_VAR_ENUM_DIRTY)
-> _CLASSNAME_EXPANSION(CLASSNAME,PREFAB_VAR_ENUM_DIRTY)
-> CLASSNAME ## _EXPANSION(PREFAB_VAR_ENUM_DIRTY)
-> CLASSNAME_EXPANSION(PREFAB_VAR_ENUM_DIRTY)

并且由于此宏的名称与原始宏的名称相同(即它是一个嵌套名称),因此我们不对其进行扩展。