c – スレッドを通したQt 5.1 QMLプロパティ

解決の目的で、私は私が持っているのと同じ問題を繰り返すTestAppを作成しました。

私のソフトウェアをQt 4.8からQt 5.1に移植しています。

私の最初のプログラムはマルチスレッドで、クラスがスレッドセーフであるという条件で、QMLを使って円滑に作業していました。しかし今、私はこのメッセージを得ます:

QObject::connect: No such slot TestApp::run() in ..\ThreadingTest\main.cpp:21
QQmlEngine: Illegal attempt to connect to TestApp(0x29cfb8) that is in a different thread than the QML engine QQmlEngine(0x2f3e0f8).

これはエラーを再現するコードです:

main.cpp:

#include <QtGui/QGuiApplication>
#include <QQmlContext>
#include <QThread>
#include "qtquick2applicationviewer.h"
#include "testapp.h"

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

    QGuiApplication app(argc, argv);

    QtQuick2ApplicationViewer viewer;

    TestApp * testapp = new TestApp();

    QThread * testappThread;

    testappThread = new QThread();

    QObject::connect(testappThread, SIGNAL(started()), testapp, SLOT(run()));

    testapp->moveToThread(testappThread);

    testappThread->start();

    viewer.rootContext()->setContextProperty("TestApp", testapp);

    viewer.setMainQmlFile(QStringLiteral("qml/ThreadingTest/main.qml"));
    viewer.showExpanded();

    out = app.exec();

    testappThread->quit();
    testappThread->wait();

    delete testapp;
    delete testappThread;

    return out;
}

testapp.h:

#ifndef TESTAPP_H
#define TESTAPP_H

#include <QObject>
#include <QString>
#include <QTimer>
#include <QReadWriteLock>

#define HELLOWORLD "Hello World !"

extern QReadWriteLock HelloWorldLock;

class TestApp : public QObject
{
    Q_OBJECT

    Q_PROPERTY(QString HelloWorld READ getHelloWorld WRITE setHelloWorld NOTIFY HelloWorldChanged)
public:
    explicit TestApp(QObject *parent = 0);

    virtual ~TestApp();

    QString getHelloWorld();

    void setHelloWorld(QString);

public slots:

    void run();

    void toggleHelloWorld();

signals:

    void HelloWorldChanged();

private:

    QString m_HelloWorld;
    QTimer * m_Timer;

};

#endif // TESTAPP_H

testapp.cpp:

#include "testapp.h"

QReadWriteLock HelloWorldLock(QReadWriteLock::Recursive);

TestApp::TestApp(QObject *parent) :
    QObject(parent)
{
    HelloWorldLock.lockForWrite();
    m_HelloWorld = HELLOWORLD;
    HelloWorldLock.unlock();

    m_Timer = new QTimer(this);

    connect(m_Timer, SIGNAL(timeout()), this, SLOT(toggleHelloWorld()));
}

TestApp::~TestApp() {
    m_Timer->stop();

    delete m_Timer;
}

QString TestApp::getHelloWorld() {
    HelloWorldLock.lockForRead();
    QString out = m_HelloWorld;
    HelloWorldLock.unlock();

    return out;
}

void TestApp::setHelloWorld(QString text) {
    HelloWorldLock.lockForWrite();
    m_HelloWorld = text;
    HelloWorldLock.unlock();

    emit HelloWorldChanged();
}

void TestApp::run() {
    m_Timer->start(1000);
}

void TestApp::toggleHelloWorld() {
    HelloWorldLock.lockForWrite();
    if(m_HelloWorld == "") {
        m_HelloWorld = HELLOWORLD;
    }
    else {
        m_HelloWorld = "";
    }
    HelloWorldLock.unlock();

    emit HelloWorldChanged();
}

main.qml:

import QtQuick 2.0

Rectangle {
    width: 360
    height: 360
    Text {
        text: TestApp.HelloWorld
        anchors.centerIn: parent
    }
    MouseArea {
        anchors.fill: parent
        onClicked: {
            Qt.quit();
        }
    }
}

私のプログラムは非常に複雑で(インターフェースと共有するためのたくさんのプロパティーとクラス)、私は私のプロパティーを接続するためだけにインターフェース・クラスを作成する必要はありません…この問題に対処するための何か提案はありますか?

ベストアンサー
Qt5でアプリケーションを自分でスレッド化する必要はありません。QML2エンジンはすでに大規模マルチスレッド化されているため、QQuickViewを起動し、必要なC部分をコンテキストに公開し、QMLファイルを設定してshow() 。それで十分です。自分でQMLスレッドを変更しようとしないでください。これは、QML1の場合よりも本当に複雑です。

転載記事の出典を記入してください: c – スレッドを通したQt 5.1 QMLプロパティ - コードログ