QObject :: connect:无法对MyClass * const类型的参数进行排队

时间:2017-02-10 15:43:15

标签: c++ qt signals-slots queued-connection

我有这样的课程:

#include <QObject>
namespace taservices
{
    class ProcessHandle : public QObject
    {
        Q_OBJECT
    public:
        ProcessHandle(const void* const processContextPointer, const QString& process_id = "", QObject *parent = 0);
        ProcessHandle();
    signals:
        void progress(const ProcessHandle* const self, const int value);
    private:
        static void registerAsMetaType();
}

我有一个信号:

void progress(const ProcessHandle* const self, const int value);

我想通过QueuedConnedtion连接它。我一直收到这条消息:

QObject::connect: Cannot queue arguments of type 'ProcessHandle*const'
(Make sure 'ProcessHandle*const' is registered using qRegisterMetaType().)

我在宣布之后注册我的课程:

Q_DECLARE_METATYPE(taservices::ProcessHandle*);

我还添加了这个静态方法,我从构造函数调用:

void ProcessHandle::registerAsMetaType()
{
    static bool called = false;
    if(!called) {
        called = true;
        qRegisterMetaType<ProcessHandle*>("taservices::ProcessHandle*");
    }
}

我也尝试注册const指针:

qRegisterMetaType<ProcessHandle*const>("taservices::ProcessHandle*const");

导致以下错误:

error C2440: 'return' : cannot convert from 'taservices::ProcessHandle *const *' to 'void *'

那么如何使我的课程与排队连接一起工作?

2 个答案:

答案 0 :(得分:3)

原来这就是你需要的:

qRegisterMetaType<ProcessHandle*>("ProcessHandle*");
qRegisterMetaType<ProcessHandle*>("ProcessHandle*const");

为了排队参数,const指针等于普通指针,因为它被复制没有改变。

答案 1 :(得分:3)

使用const值参数发出信号毫无意义。 const的唯一要点是防止实现行为不端并修改值,暗示不应该以某种方式修改值(为什么不呢?这是你泄漏到接口中的实现细节!)。信号的代码是由moc生成的,如果这些代码行为不端,你已经遇到了更大的问题。

您的信号应具有以下声明:

Q_SIGNAL void progress(const ProcessHandle* self, int value);

插槽可以自由地使用const参数。就Qt而言,最内部的constness不是签名的一部分 - 它被有效地剥离了。

您无需注册类型。当您通过使用新的connect语法让connect访问类型时,它会自动完成。

示例:

// https://github.com/KubaO/stackoverflown/tree/master/questions/const-slot-arg-42163294
#include <QtCore>

struct ProcessHandle {};

struct Object : QObject {
    int counter = 0;
    Q_SIGNAL void newValue(const ProcessHandle*, int val);
    Q_SLOT void useValue(const ProcessHandle* const ph, const int val) {
        qDebug() << ph << val;
        Q_ASSERT(ph == nullptr && val == 42);
        ++counter;
    }
    Q_OBJECT
};

int main(int argc, char ** argv) {
    QCoreApplication app{argc, argv};
    Object obj;
    QObject::connect(&obj, &Object::newValue, &obj, &Object::useValue, Qt::QueuedConnection);
    QObject::connect(&obj, &Object::newValue, &app, &QCoreApplication::quit, Qt::QueuedConnection);
    emit obj.newValue(nullptr, 42);
    app.exec();
    Q_ASSERT(obj.counter == 1);
}

#include "main.moc"