使用 QT 进程或线程来运行函数?

问题描述

我想使用不同的线程或 QProcess 或更有效并提供良好性能的东西来运行几个函数。我正在尝试为校准器构建一个 gui,只要我按下校准按钮,它就需要校准。对于校准,我有一些功能可以做到这一点。但是,执行此校准过程需要一些时间。在我的校准完成之前,我的 gui 一直没有响应。所以我在这里要做的是我想在一些并行进程或线程或其他一些东西中运行该函数

#include "widget.h"
#include "ui_widget.h"

#include <iostream>
#include <sstream>

#include <QtConcurrent/QtConcurrent>
#include <QFuture>

using namespace Eigen;

Widget::Widget(QWidget *parent) :
    QWidget(parent),ui(new Ui::Widget)
{
    ui->setupUi(this);


    ui->lineEdit->setReadOnly(true);
        
}

Widget::~Widget()
{
    delete ui;
}

void Widget::on_pushButton_pressed()
{
    ui->plainTextEdit->appendHtml("<div style='color: green;'> Calibrating .....  </div>");

    QString body = ui->comboBox->currentText();
    int body = body.toInt();
    /*
    I want to run calib function here in separate thread or QtConcurrent so my gui will stay resposive even my function takes some time close to 1 min to do calibration
    
    */

}

void Widget::printReceivedBody(int x)
{
    
}

    

    
void Widget::Calib(int x)
{

    printReceivedBody(int x)
}

我在堆栈和 qt 中检查了一些问题,但无法解决我的问题。我发现了几个问题,但正在尝试运行可执行文件。我也可以通过可执行文件来做,但我想尝试上述方法而不是使用可执行文件

我去了 qt Documentaion 但那个文档让我更加困惑:-(

有人可以建议我怎么做吗?

解决方法

A QProcess 用于运行外部进程(如果可能更清楚,则为可执行文件)。以我的拙见,这不是您所需要的。

您可能需要的是使用 QThread 来运行您的校准功能。

您已经看过文档,让我们为您提供一些如何在 QT 中使用 QThread 执行函数的示例。

// execute a lambda
void MainWindow::onButton1Click()
{
    qDebug()<< "clicked";
    qDebug() << " the main thread id = " << QThread::currentThread();

    QThread* l_thread = QThread::create([&]()
    {
        qDebug() << "Running Thread " << QThread::currentThreadId() << " to emit signal only ";
        //emit dummy signal (for instance to refresh GUI
        //emit sigShowHide( !this->ui->pushButton_2->isVisible());
    });
    l_thread->start();
}

// example of an external function to execute

void test(int value1,int &value2)
{
    value2 = value1 + 1;
    return;
}

void MainWindow::onButton2Click()
{
    
    auto func = std::bind(test,value1,std::ref(value2));
    QThread* qthread = QThread::create(std::bind(test,std::ref(value2)));
    qthread ->start();

}


// execute a method defined in MainWindow
void MainWindow::count()
{
    qDebug()<< "Counting";
}


void MainWindow::onButton3Click()
{
     ui->label->setText("Starting to count");
     auto function = std::bind(&MainWindow::count,this);
     QThread* l_thread = QThread::create(function);
     l_thread->start();
}

更新: 至于你(在更新的问题中),你可以像下面这样运行校准

void Widget::on_pushButton_pressed()
{
    ui->plainTextEdit->appendHtml("<div style='color: green;'> Calibrating .....  </div>");

    QString body = ui->comboBox->currentText();
    int body = body.toInt();
    /*
    I want to run calib function here in separate thread or QtConcurrent so my gui will stay resposive even my function takes some time close to 1 min to do calibration
    */
    // LIKE THIS
    // for instance you can to call Widget::Calib with the arugment int x=15
    int x =15;
    auto func = std::bind(&Widget::Calib,this,x));
    QThread* qthread = QThread::create(func);
    qthread ->start();
    
    // if you want to wait for the QThread to finish before continuing,you can add 
    // qthread.wait()
}

仅供参考,这里是使用 QProcess 启动 Windows 命令行 cmd.exe 并执行带有参数的一些 copy 命令的示例

#include <QCoreApplication>
#include <Qdebug>
#include <QObject>
#include <QProcess>

int main(int argc,char *argv[])
{
    QCoreApplication app(argc,argv);
    QProcess* processus = new QProcess();
    QStringList args; 
    QString path("/path_accentué");
       
    args << "/C" << "copy" << "/toto/sfx.exe" << path;// "F:\\path_accentué";
    processus->start("cmd",args);  

    if (!processus->waitForStarted())
    {
        qDebug() << "Could not launch the process";
    }
    processus->write(s.c_str());
    if (!processus->waitForFinished(-1))
    {
        qDebug() << "Finished";
    }
    delete processus;
    return app.exec();
}
,

为了保持您的 GUI 响应,需要在后台线程中执行繁重的处理。

您可以使用标准库运行线程:

#include <thread>
...
std::thread t(Calib,42);
...
t.join();

显然,不要在你的主线程中同步加入,否则你会失去它的所有好处。

为了显示结果,Qt 不允许从后台线程修改 GUI,因此您必须将信号入队,即将 connect 配置为 Qt::QueuedConnection

Qt signals (QueuedConnection and DirectConnection)