#qt #class #signals-slots #unresolved-external
#qt #класс #сигналы-слоты #неразрешенный-внешний
Вопрос:
Я создал приложение QtGui с помощью qt designer и добавил к нему простой класс. Несмотря на то, что этот класс наследуется от класса QObject и имеет макрос Q_OBJECT в закрытом разделе, код не запускается, и я столкнулся с этой ошибкой:
LNK2001 неразрешенный внешний символ «общедоступный: виртуальная структура QMetaObject const * __cdecl Sender::metaObject(void)const » (?metaObject@Sender@@UEBAPEBUQMetaObject@@XZ) QtGuiApplication1
заголовок класса:
#ifndef SENDER_H
#define SENDER_H
#include <QObject>
class Sender : public QObject
{
Q_OBJECT
public:
explicit Sender(QObject *parent = 0);
void fireSignal();
signals:
void foo(const QStringamp; arg);
};
#endif // SENDER_H
класс cpp:
#include "Sender.h"
Sender::Sender(QObject *parent) :
QObject(parent)
{
}
void Sender::fireSignal()
{
emit foo("aa");
}
Заголовок графического интерфейса:
#pragma once
#include <QtWidgets/QMainWindow>
#include "ui_QtGuiApplication1.h"
class QtGuiApplication1 : public QMainWindow
{
Q_OBJECT
public:
QtGuiApplication1(QWidget *parent = Q_NULLPTR);
private:
Ui::QtGuiApplication1Class ui;
};
Графический интерфейс cpp:
#include "QtGuiApplication1.h"
#include <QPainter>
QtGuiApplication1::QtGuiApplication1(QWidget *parent)
: QMainWindow(parent)
{
ui.setupUi(this);
}
Когда я комментирую макрос Q_OBJECT и emit foo("aa");
это работает.
Где я допускаю ошибки? Спасибо.
отредактировано: прикрепить moc_Sender.cpp
/****************************************************************************
** Meta object code from reading C file 'Sender.h'
**
** Created by: The Qt Meta Object Compiler version 67 (Qt 5.11.1)
**
** WARNING! All changes made in this file will be lost!
*****************************************************************************/
#include "../../Sender.h"
#include <QtCore/qbytearray.h>
#include <QtCore/qmetatype.h>
#if !defined(Q_MOC_OUTPUT_REVISION)
#error "The header file 'Sender.h' doesn't include <QObject>."
#elif Q_MOC_OUTPUT_REVISION != 67
#error "This file was generated using the moc from 5.11.1. It"
#error "cannot be used with the include files from this version of Qt."
#error "(The moc has changed too much.)"
#endif
QT_BEGIN_MOC_NAMESPACE
QT_WARNING_PUSH
QT_WARNING_DISABLE_DEPRECATED
struct qt_meta_stringdata_Sender_t {
QByteArrayData data[4];
char stringdata0[16];
};
#define QT_MOC_LITERAL(idx, ofs, len)
Q_STATIC_BYTE_ARRAY_DATA_HEADER_INITIALIZER_WITH_OFFSET(len,
qptrdiff(offsetof(qt_meta_stringdata_Sender_t, stringdata0) ofs
- idx * sizeof(QByteArrayData))
)
static const qt_meta_stringdata_Sender_t qt_meta_stringdata_Sender = {
{
QT_MOC_LITERAL(0, 0, 6), // "Sender"
QT_MOC_LITERAL(1, 7, 3), // "foo"
QT_MOC_LITERAL(2, 11, 0), // ""
QT_MOC_LITERAL(3, 12, 3) // "arg"
},
"Senderfooarg"
};
#undef QT_MOC_LITERAL
static const uint qt_meta_data_Sender[] = {
// content:
7, // revision
0, // classname
0, 0, // classinfo
1, 14, // methods
0, 0, // properties
0, 0, // enums/sets
0, 0, // constructors
0, // flags
1, // signalCount
// signals: name, argc, parameters, tag, flags
1, 1, 19, 2, 0x06 /* Public */,
// signals: parameters
QMetaType::Void, QMetaType::QString, 3,
0 // eod
};
void Sender::qt_static_metacall(QObject *_o, QMetaObject::Call _c, int _id, void **_a)
{
if (_c == QMetaObject::InvokeMetaMethod) {
Sender *_t = static_cast<Sender *>(_o);
Q_UNUSED(_t)
switch (_id) {
case 0: _t->foo((*reinterpret_cast< const QString(*)>(_a[1]))); break;
default: ;
}
} else if (_c == QMetaObject::IndexOfMethod) {
int *result = reinterpret_cast<int *>(_a[0]);
{
using _t = void (Sender::*)(const QString amp; );
if (*reinterpret_cast<_t *>(_a[1]) == static_cast<_t>(amp;Sender::foo)) {
*result = 0;
return;
}
}
}
}
QT_INIT_METAOBJECT const QMetaObject Sender::staticMetaObject = {
{ amp;QObject::staticMetaObject, qt_meta_stringdata_Sender.data,
qt_meta_data_Sender, qt_static_metacall, nullptr, nullptr}
};
const QMetaObject *Sender::metaObject() const
{
return QObject::d_ptr->metaObject ? QObject::d_ptr->dynamicMetaObject() : amp;staticMetaObject;
}
void *Sender::qt_metacast(const char *_clname)
{
if (!_clname) return nullptr;
if (!strcmp(_clname, qt_meta_stringdata_Sender.stringdata0))
return static_cast<void*>(this);
return QObject::qt_metacast(_clname);
}
int Sender::qt_metacall(QMetaObject::Call _c, int _id, void **_a)
{
_id = QObject::qt_metacall(_c, _id, _a);
if (_id < 0)
return _id;
if (_c == QMetaObject::InvokeMetaMethod) {
if (_id < 1)
qt_static_metacall(this, _c, _id, _a);
_id -= 1;
} else if (_c == QMetaObject::RegisterMethodArgumentMetaType) {
if (_id < 1)
*reinterpret_cast<int*>(_a[0]) = -1;
_id -= 1;
}
return _id;
}
// SIGNAL 0
void Sender::foo(const QString amp; _t1)
{
void *_a[] = { nullptr, const_cast<void*>(reinterpret_cast<const void*>(amp;_t1)) };
QMetaObject::activate(this, amp;staticMetaObject, 0, _a);
}
QT_WARNING_POP
QT_END_MOC_NAMESPACE
Комментарии:
1. Как вы создаете свой пример? Вероятно, вам нужно запустить
moc
в заголовке сQ_OBJECT
макросом и построить и связать в результирующем сгенерированном коде. Самый простой способ сделать это с помощьюqmake
илиcmake
.2. спасибо за ваше время. Я не понял, что вы имели в виду. Я использовал Visual studio 2017 и qt designer. Как я мог бы запустить moc?
3. Щелкните правой кнопкой мыши на вашем проекте, выберите Преобразовать проект в проект Qt VS Tools . Он автоматически сгенерирует
.moc
файлы. Если вы не можете его найти, нажмите, чтобы Преобразовать Project в сгенерированный QMake проект , затем снова выполните поиск Преобразовать Project в Qt VS Tools Project4. @arsdever Я преобразовал проект в Qt VS Tools… Но проблема все еще остается
5. Затем, пожалуйста, убедитесь, что ваш moc_Sender.cpp находится в вашем разделе проекта GeneratedFiles и, пожалуйста, также вставьте содержимое moc_Sender.cpp здесь. Если вы не найдете moc_Sender.cpp в любом месте щелкните правой кнопкой мыши на Sender.h и убедитесь, что он использует пользовательский инструмент сборки.