3 回答
TA贡献1844条经验 获得超8个赞
每次向Q_OBJECT宏添加新调用时,都需要再次运行qmake。您所指的vtable问题与此直接相关。
只需运行qmake,您应该好好假设您的代码中没有其他问题。
TA贡献1876条经验 获得超7个赞
我已经看到很多方法来解决这个问题,但没有解释为什么会发生这种情况,所以这里就是这样。
当编译器看到具有虚函数的类(直接声明或继承)时,它必须为该类生成vtable。由于类通常在头文件中定义(因此出现在多个翻译单元中),因此问题在于放置vtable的位置。
通常,可以通过在定义类的每个TU中生成vtable来解决问题,然后让链接器消除重复。由于ODR在每次出现时要求类定义相同,因此这是安全的。但是,它也会降低编译速度,膨胀对象文件,并要求链接器执行更多工作。
因此,作为优化,编译器将在可能的情况下选择特定的TU来放入vtable。在通用的C ++ ABI中,这个TU 是实现类的关键功能的那个,其中关键功能是第一个在类中声明但未定义的虚拟成员函数。
在Qt类的情况下,它们通常以Q_OBJECT宏开头,并且该宏包含声明
virtual const QMetaObject *metaObject() const;
因为它是宏中的第一个虚函数,所以它通常是该类的第一个虚函数,因此也是它的关键函数。因此,编译器不会在大多数TU中发出vtable,只会执行实现的vtable metaObject
。此函数的实现moc
在处理标头时自动写入。因此,您需要moc
处理标头以生成新的.cpp文件,然后在编译中包含.cpp文件。
因此,当您有一个定义了一个QObject
派生类的新标头时,您需要重新运行qmake
以便它更新您的makefile以moc
在新标头上运行并编译生成的.cpp文件。
- 3 回答
- 0 关注
- 1352 浏览
添加回答
举报