没有什么可以帮助解决 Qt 中的“对 vtable 的未定义引用”

ban*_*a36 2 qt vtable qobject

我无法构建此错误“未定义对 CustomUndoStack 的 vtable 的引用”

这是代码:

class CustomUndoStack : public QObject
{
    Q_OBJECT

public:

};


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

    CustomUndoStack uStack;

    return qCoreApplication.exec();
}
Run Code Online (Sandbox Code Playgroud)

这是 .pro 文件:

QT += widgets

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

DEFINES += QT_DEPRECATED_WARNINGS

SOURCES += main.cpp
Run Code Online (Sandbox Code Playgroud)

我已经手动删除了构建目录,我已经以所有可能的方式尝试了“全部清理”和“重新构建全部”,我已经运行了 qmake 100500 次,我已经更改了 CustomUndoStack 界面超过 9000 次,但没有任何帮助。

问题出在哪里?

UPD 回答评论:如果我删除 Q_OBJECT 行,那么我就不能这样做。它说“没有这样的信号 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;
};
Run Code Online (Sandbox Code Playgroud)

Dmi*_*try 5

您的问题是由于您在文件中声明了类子类QObjectQ_OBJECT在其中包含宏而引起的.cpp;该宏除其他内容外还包含虚拟方法的声明:

virtual const QMetaObject *metaObject() const;
Run Code Online (Sandbox Code Playgroud)

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

正确的解决方案是将类的声明移动到单独的头文件中,并将该头添加到您的项目中。然而,还有另一种解决方案,虽然有点 hacky,但如果您出于某种原因绝对需要将类的声明留在文件中,则可以使用它.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();
}
Run Code Online (Sandbox Code Playgroud)

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