首页 > 解决方案 > Qt 信号以一种奇怪的方式发出

问题描述

我有以下按钮代码:

emit busySignal();
QString program = "cmd";
QStringList arguments;
arguments << "";
QProcess *myProcess = new QProcess(this);
myProcess->start(program, arguments);
myProcess->write("cd C:\\Qt\\5.11.2\\mingw53_32\\bin\\\n");
myProcess->waitForBytesWritten();
myProcess->write("windeployqt C:\\Users\\BRSLV\\Desktop\\qml_collect_project\n");
myProcess->waitForBytesWritten();
myProcess->waitForFinished();
myProcess->close();
emit readySignal();

并遵循 qml 代码:

Model
{
    id : model
    onBusySignal: busy.running = true
    onReadySignal: busy.running = false
}
BusyIndicator {
    id: busy
    running: false

}

问题是 - 由于某种原因,信号不能像我期望的那样工作:busySignal() 只有在 QProcess 完成后才会影响 qml。有人可以告诉我如何解决吗?谢谢你。

标签: c++qtqmlqt5qprocess

解决方案


waitForXXX 方法是阻塞的,所以会冻结GUI,阻止执行信号传输等异步任务,所以解决方法是用信号知道什么时候通知任务完成。

另一方面,您的代码可以使用setWorkingDirectory()该替换cd命令来减少。

主文件

#include <QGuiApplication>
#include <QProcess>
#include <QQmlApplicationEngine>

class ProcessManager: public QObject
{
    Q_OBJECT
public:
    ProcessManager(QObject *parent=nullptr):
        QObject(parent)
    {
        connect(&m_process, QOverload<int>::of(&QProcess::finished), this, &ProcessManager::readySignal);
    }
    Q_INVOKABLE void start_process(){
        emit busySignal();
        const QString program = "cmd";
        m_process.setWorkingDirectory("C:\\Qt\\5.11.2\\mingw53_32\\bin");
        m_process.start(program, {"windeployqt", "C:\\Users\\BRSLV\\Desktop\\qml_collect_project"});
    }
signals:
    void busySignal();
    void readySignal();
private:
    QProcess m_process;
};

int main(int argc, char *argv[])
{
    QCoreApplication::setAttribute(Qt::AA_EnableHighDpiScaling);
    qmlRegisterType<ProcessManager>("com.utils", 1, 0, "ProcessManager");
    QGuiApplication app(argc, argv);

    QQmlApplicationEngine engine;
    engine.load(QUrl(QStringLiteral("qrc:/main.qml")));
    if (engine.rootObjects().isEmpty())
        return -1;
    return app.exec();
}
#include "main.moc"

main.qml

import QtQuick 2.9
import QtQuick.Window 2.2
import QtQuick.Controls 2.5

import com.utils 1.0

Window {
    visible: true
    width: 640
    height: 480
    title: qsTr("Hello World")
    ProcessManager{
        id: pm
        onBusySignal: busy.running = true
        onReadySignal:  busy.running = false
    }
    BusyIndicator {
        id: busy
        running: false
    }
    Button{
        anchors.top: busy.bottom
        text: "start process"
        onClicked: pm.start_process()
    }
}

推荐阅读