本文主要是介绍QtConcurrent::run操作界面ui的注意事项,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
先说结论:QtConcurrent::run启动的耗时处理函数,不允许处理ui界面对象,如控件,如进度条等等!
QtConcurrent::run非常好用,胜过QThead的两种方式(run和moveToThread),例如下面是非常直观和简单的使用方式:
QT += concurrent#include <QtConcurrent>
#include <QThread>void MainWindow::doSomeWork()//死循环操作,代替原有代码的函数功能
{int i=0;while(myEnable){i++;qDebug()<<i;QThread::msleep(10);}
}void MainWindow::myStart()//启动函数是需要在原有代码基础上增加
{myEnable=1;QtConcurrent::run(this,&MainWindow::doSomeWork);//多线程执行死循环启动,可以带参数void MainWindow::myExit()
{myEnable=0;//
}
但是,但是,上述的MainWindow::doSomeWork函数中不能访问ui对象,会引起程序崩溃!!原因后面解释。先说解决办法:
一种方法是,MainWindow::doSomeWork能读写简单类型的数值变量,int,float类型,则可以借助这些简单类型变量做个过渡中转一下。既然是ui界面对象,是给人看的,延时几十个ms,人的眼睛不会有感觉,因此,开一个定时器,轮询这些简单的数值int、float变量,在定时器里去更新ui界面对象,这是最简单的方法;
另一种方法是,在调用QtConcurrent::run(this,&MainWindow::doSomeWork)的地方,检查返回值,开一个循环等待完成。在循环里完成对界面ui的操作,如下面在按钮点击事件中:
void MainWindow::on_pushButton_clicked()
{ui->progressBar->setRange(0, 100); //使用进度条QFuture<void> future = QtConcurrent::run(this,&MainWindow::doSomeWork);//多线程执行死循环启动,可以带参数while(!future.isFinished()){ui->progressBar->setValue(m_curVal); //设置进度条当前值,m_curVal在doSomeWork中被更新QApplication::processEvents(QEventLoop::AllEvents, 100); //避免ui挂死}
}
上面依然是第一种方式的变种,只不过在循环里,借助了QApplication::processEvents(QEventLoop::AllEvents, 100),避免将ui挂死。
小结上述两种方法,第一种其实就是“共享全局变量”的变形;第二种是把循环直接写在ui的按钮事件里,让ui保持“灵活”;
再回到上述的:MainWindow::doSomeWork函数中不能访问ui对象,会引起程序崩溃!!
原因是,QtConcurrent::run本质是开启了另外一个子线程,该子线程和ui线程不相同,可以用:
qDebug()<<"当前线程 "<<QThread::currentThread();
分别加入到MainWindow的构造函数、on_pushButton_clicked()函数、doSomeWork()函数里验证,会发现,不在同一个线程里。这是qt的“跨线程”问题。
因为涉及到“跨线程”问题,总的来说,QtConcurrent::run并不好用,如果开启的线程里,要读写套接字tcp/udp、串口,是有问题的。
这篇关于QtConcurrent::run操作界面ui的注意事项的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!