在 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
生成的声明相对应的自动生成代码宏。
我无法构建此错误 "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
生成的声明相对应的自动生成代码宏。