在 Qt 中没有任何帮助解决 "Undefined reference to vtable"

Nothing helped to solve "Undefined reference to vtable" in Qt

我无法构建此错误 "undefined reference to vtable for CustomUndoStack"

代码如下:

class CustomUndoStack : public QObject
{
    Q_OBJECT

public:

};


int main(int argc, char *argv[])
{
    QCoreApplication qCoreApplication(argc, argv);

    CustomUndoStack uStack;

    return qCoreApplication.exec();
}

这是 .pro 文件:

QT += widgets

CONFIG += c++14 console
CONFIG -= app_bundle

DEFINES += QT_DEPRECATED_WARNINGS

SOURCES += main.cpp

我已经手动删除了构建目录,我已经尝试了 "Clean All" 和 "Rebuild All" 所有可能的方式,我已经 运行 qmake 100500 次,我已经改变了CustomUndoStack 接口超过 9000 次,但没有任何帮助。

哪里出了问题?

UPD 回答评论: 如果我删除 Q_OBJECT 行,那么我不能这样做。它说 "No such signal QObject::undoSignal()"

class CustomUndoStack : public QObject
{
//    Q_OBJECT

public:
    CustomUndoStack() {
        connect(this, SIGNAL(undo_signal()), &undoStack, SLOT(undo()));
    }
signals:
    void undo_signal();
private:
    QUndoStack undoStack;
};

你的问题是由于你声明了一个 class subclassing QObject 并且在 [=] 中有 Q_OBJECT 宏14=] 文件;该宏除其他内容外,还包含虚方法的声明:

virtual const QMetaObject *metaObject() const;

qmake 工具仅处理头文件以定位包含 Q_OBJECT 宏的 QObject 子 classes 以自动生成与 [= 产生的声明对应的实际代码13=]宏。在您的情况下,qmake 不处理 main.cpp 文件,因此它包含未实现的虚函数 - 因此您收到链接器错误。

正确的解决方案是将 class 的声明移动到一个单独的头文件中,并将该头文件添加到您的项目中。然而,还有另一种解决方案,它有点老套,但如果您出于某种原因绝对需要将 class 的声明留在 .cpp 文件中,则可以使用它:

class CustomUndoStack : public QObject
{
    Q_OBJECT
public:
    CustomUndoStack(QObject * parent = 0) : QObject(parent) {}
};

#include "main.moc"

int main(int argc, char *argv[])
{
    QCoreApplication qCoreApplication(argc, argv);

    CustomUndoStack uStack;

    return qCoreApplication.exec();
}

注意 #include "main.moc" 行。此行的存在强制 qmake 实际处理来自 .cpp 文件的声明并生成所需的 .moc 文件,其中包含与 Q_OBJECT 生成的声明相对应的自动生成代码宏。