Eu só quero iniciar um aplicativo QT simples. Eu tenho main.cpp
, mainwindow.h
e mainwindow.cpp
. Isso funciona, no entanto, quando copio e colo o código de mainwindow.h
e mainwindow.cpp
para main.cpp
, recebo erros misteriosos do vinculador.
O QUE FUNCIONA
// mainwindow.h
#ifndef MAINWINDOW_H
#define MAINWINDOW_H
#include <QtWidgets/QMainWindow>
QT_BEGIN_NAMESPACE
namespace Ui {
class MainWindow;
}
QT_END_NAMESPACE
class MainWindow : public QMainWindow
{
Q_OBJECT
public:
MainWindow(QWidget* parent = nullptr);
~MainWindow();
private:
Ui::MainWindow* ui;
};
#endif
//mainwindow.cpp
#include "mainwindow.h"
#include "ui_mainwindow.h"
MainWindow::MainWindow(QWidget* parent) : QMainWindow(parent)
{
ui = new Ui::MainWindow;
ui->setupUi(this);
}
MainWindow::~MainWindow() { delete ui; }
//main.cpp
#include "mainwindow.h"
#include <QApplication>
int main(int argc, char **argv)
{
QApplication a(argc, argv);
MainWindow w;
w.show();
return a.exec();
}
O QUE NÃO FUNCIONA
//main.cpp
#include <QApplication>
#include "ui_mainwindow.h"
#include <QtWidgets/QMainWindow>
QT_BEGIN_NAMESPACE
namespace Ui {
class MainWindow;
}
QT_END_NAMESPACE
class MainWindow : public QMainWindow
{
Q_OBJECT
public:
MainWindow(QWidget* parent = nullptr);
~MainWindow();
private:
Ui::MainWindow* ui;
};
MainWindow::MainWindow(QWidget* parent) : QMainWindow(parent)
{
ui = new Ui::MainWindow;
ui->setupUi(this);
}
MainWindow::~MainWindow() { delete ui; }
int main(int argc, char **argv)
{
QApplication a(argc, argv);
MainWindow w;
w.show();
return a.exec();
}
Isso me dá os seguintes erros durante a vinculação:
/usr/bin/ld: /tmp/ccX909s9.ltrans1.ltrans.o: in function `MainWindow::~MainWindow()':
<artificial>:(.text+0x3): undefined reference to `vtable for MainWindow'
/usr/bin/ld: /tmp/ccX909s9.ltrans1.ltrans.o: in function `main':
<artificial>:(.text.startup+0x4a): undefined reference to `vtable for MainWindow'
Depois de algumas pesquisas, esse erro muito detalhado significa que há uma função virtual em algum lugar que não está implementada. Estou realmente lutando para ver como isso importa aqui.
Para referência, sou tradicionalmente um programador C e, da minha perspectiva, esses programas deveriam ser idênticos. Eu realmente adoraria entender que jargão mágico o compilador está procurando neste momento.
Como foi esclarecido nos comentários, os 2 programas não são idênticos: no primeiro caso, nada cria os métodos ocultos dentro da
Q_OBJECT
macro, alguns dos quais são virtuais (metaObject
é o mais importante aí). Essa deveria ser a função domoc
(compilador de metaobjeto).O que você obtém são métodos virtuais declarados, mas não implementados.
Ao contrário do acima, sua abordagem de cabeçalho + arquivos de origem faz com que o arquivo de cabeçalho seja moc'd, o que cria um arquivo de origem separado com os métodos necessários. No Qt Creator ou no Visual Studio com a extensão Qt VS Tools instalada, tudo acontece automaticamente; nenhuma ação sua é necessária para configurar as etapas (pode ser o caso em outros ambientes de desenvolvimento).
No Visual Studio Vanilla (o único ambiente que usei Qt com configuração manual de tudo), você teria que configurar manualmente uma etapa de construção personalizada e, em seguida, incluir o arquivo fonte gerado para que ele fosse compilado. Configurar tudo manualmente é um pouco trabalhoso, mas isso permitiria que você identificasse facilmente o erro.
Obs: Como falei nos comentários, a
Q_OBJECT
macro não é obrigatória para subclasses deQObject
. Se você não declarar nada que possa modificar o metaobjeto da classe (sinais, slots, propriedades, interface, ...), então a macro poderá ser ignorada.AFAICT, o código da sua pergunta não precisa
Q_OBJECT
, portanto, o snippet main.cpp-only da sua pergunta deve funcionar depois de simplesmente removê-lo.