【问题标题】:using QT Process or threads to run functions?使用 QT 进程或线程来运行函数?
【发布时间】:2021-05-13 05:29:32
【问题描述】:

我想使用不同的线程或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,但该文档让我更加困惑 :-(

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

【问题讨论】:

  • 您不想运行可执行文件,但想知道是否应该使用 QProcess?这就是 QProcess 所做的一切。因此,如果您不想运行可执行文件,则可以排除 QProcess。查找QThread 的示例。它们数量众多。
  • QtConcurrent::run 是另一种选择。
  • 几天前我回答了一个类似的question

标签: c++ qt qthread qprocess qtconcurrent


【解决方案1】:

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

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

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

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

为了显示结果,Qt不允许从后台线程修改GUI,所以你必须将一个信号入队,即配置connectQt::QueuedConnection

Qt signals (QueuedConnection and DirectConnection)

【讨论】:

  • 不要使用join(),因为它会阻塞主线程并因此冻结GUI
  • @Adrian Maire。非常感谢。将对此进行研究。但我更愿意在此使用 qt 功能,例如 QtConcurrent 或其他东西
【解决方案2】:

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,value1, 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 并执行带有参数的somme copycommands 的示例

#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();
}

【讨论】:

  • 使用带有跨平台 API 的 Windows 路径有什么意义?始终使用“/”分隔符进行编码,并且没有字母。尤其是使用 BSD 头像。
  • 感谢您的建议。我以前做过很多次的例子中的一个糟糕的复制粘贴。已更正
  • 如果您想查看和测试,我已根据您的案例更新了我的回复
  • 您至少需要在 QT5.10 上。在文档中,您应该看到:“此功能是在 Qt 5.10 中引入的。”另外,确保在你的 QT 项目文件(.pro 文件)中有这个子句 QT +=core 如果已经存在,添加“core”
  • 对不起,如果你没有至少 QT5.10,你不会有QThread::create 它,如果我没记错的话,std::bind 是由 C++11 提供的,所以你必须与 c++11 兼容性编译。 QThread::create 的替代方法是使用工作类并将其与 QThread (mayaposch.wordpress.com/2011/11/01/…) 相关联
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2017-05-31
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多