最开始使用Qt时就遇到过QT Gui失去响应的问题,我是用多线程的方式解决的,然而通常来说,多线程是会降低程序的运行速度。

之后,在使用QSqlQuery::execBatch()函数时,Qt Gui 又失去响应,虽然多线程可以解决,但是如果能用单线程很好解决的,最好不要用到多线程,因为多线程不仅容易拖慢程序的速度,编程及维护的难度也更大,能用简单方法解决的,就不要用复杂的方法。

于是我再次搜索资料,期望在解决方案的选择与解决步骤上,能够得到一个全面而又细致的总结。


Witold Wysota 的文章https://doc.qt.io/archives/qq/qq27-responsive-guis.html#performinglongoperations 总结的非常不错。

Jason Lee的翻译: http://blog.csdn.net/jasonblog/article/details/5568589

所以本文是在此文基础上的部分翻译、理解与二次总结。总之,有删减,有补充,所以没写 '转' 字。


一、问题的来源与分析

首先,我们要知道 “为什么Qt Gui 会停止响应?”。简明扼要的说就是:长时间的密集处理或等待阻塞了Qt的事件循环,应用程序不能响应来自窗口系统的事件请求(《C++ Gui Qt4》 P135中有描述)。   那么多长算长呢?一秒钟算长,两秒钟太长。

其次,“ 何种情形下会发生该问题? ”。可分为两种情形:

第一,长时间按顺序执行的密集运算,全部计算结束后才能继续执行,如快速傅立叶变换。

第二,“ 触发 ”了某项操作,该操作完成后才能进行“ 下一步 ”, 所以这里描述的是异步操作,如保存文件操作,服务器等待连接、网络下载等。详细见附注(1).

私以为两种情形并无明显的概念上的区分,本质是一样的,但两种情形有不同的处理方法,特别是第二种情形, 在Qt框架下 ,用Qt的信号和槽机制往往可以解决阻塞问题,如QTcpServer::newConnection信号通知连接的到来,QIODevice::bytesWritten()与 QIODevice::readyRead()通知文件的读写,它们都是以非阻塞的形式实现相关功能的利器。 而第一种情形,不仅所有的事件循环停止了,信号和槽也暂时被忽视。我们将针对以上两种情形寻找解决方案。

最后,我们考虑是否可以把这个造成 Qt Gui 停止响应的罪魁祸首大卸八块,即把他拆分成一个个小块,如果可以拆分,那么每块之间是依赖还是独立,如果独立那问题好办,放在不同的位置独立运作,否则,我们只能同步的执行,而最差的结果是——根本无法拆分!!

总之,考虑以上信息差异,执行不同的解决方案。


二、解决方案

Manual event processing(人工执行事件)

保持事件循环有一种最基本的方法——让程序去处理 悬挂事件好了 ,处理完了再回来继续我的后续运算,要做到这一点,就要在我的运算代码中间加上处理事件的代码,这句代码就是 QCoreApplication::processEvents();,只要该句代码能够周期性的被执行,就能保持Qt Gui的响应。

//代码来源于上述链接所指向文章

  1. for (int i = ; i <= sqrt(x) && isPrime; i += ) {
  2. label->setText(tr("Checking %1...").arg(i));
  3. if (x % i == )
  4. isPrime = false;
  5. QCoreApplication::processEvents();
  6. if (!pushButton->isChecked()) {
  7. label->setText(tr("Aborted"));
  8. return;
  9. }
  10.  
  11. }

部分翻译(略)——可查看 Jason Lee的翻译

该方案除了 具有Witold Wysota文中 所提到缺点之外,《C++ Gui Qt4》P135中还提到,用户可能会在应用程序还在执行某种操作时,或者关闭了主窗口,或者通过界面再次触发相同操作,这样就会产生不可预料的后果,如 一个保存文件对话框,用户单击save按钮后,程序开始磁盘文件的写入操作,该操作还未完成时,用户再次单击了关闭按钮,或者再次单击save按钮。书中给出的解决办法是将 qApp->processEvents()替换为qApp-> processEvents(QEventLoop::ExcludeUserInputEvents),以告诉Qt忽略鼠标事件和键盘事件。

Using a Worker Thread(使用任务线程)

翻译(略)—— Jason Lee的翻译

除了 Witold Wysota 文中所说的重新实现QThread类之外,还可以使用QObject::moveToThread(QThread *thread)函数,将

进行复杂运算的对象移入子线程中运行,前提是子线程不能够有父对象,否则无法移入子线程。示例如下:

  1. QThread *thread = new QThread(this);
  2. MyComputation *computation = new MyComputation();//负责密集运算的对象
  3. computation->moveToThread(thread);
  4. connect(thread, SIGNAL(started()), computation ,SLOT( compute()) ); //compute()为computation的运算函数
  5. thread->start();

需注意的是,将 computation 对象移入子线程后, 依旧不可直接调用  computation 对象 compute()函,应该调用线程对象的start()函数,发出started()信号触发 computation 对象的运算操作,否则依旧会阻塞主线程。

Waiting in a Local Event Loop(在本地事件循环中等待)

翻译(略)—— Jason Lee的翻译

注解:如文章开头所说,“等待异步事件完成”,也就是说这种方法是针对异步事件而设计的,异步事件执行过程中会不断发送信号,我们根据该信号决定程序接下来的行为,包括人工执行事件。而  Manual event processing 适用于顺序执行的操作 。

Solving a Problem Step by Step(分步骤解决问题)

翻译(略)

如前文所说,如果一个复杂操作可以拆分为独立的子操作,那么拆分应该是最好的解决办法。至于如何拆分,可以通过阅读《重构》这本书来学习。

Parallel Programming
翻译(略)


三、总结

前面我提到过,我是用queryBatch()函数导致了Qt Gui无法响应的,最后我选择了 Using a Worker Thread这种方法。queryBatch()是一个操作数据库的批处理函数,非常便利,但它是顺序执行的,我无法用异步方式来处理它,函数内部是不可见的,也无法人工执行事件或者拆分,最后只能使用子线程来执行它了,这就是为便利所付出的代价吧。不同情况有不同的解决方案,认清自己的问题很重要。


四、附注

(1) “ 触发 ”了某项操作,该操作完成后才能进行“ 下一步 ”,但实际上对于“触发”这个行为本身而言,它的职责已经完成了,而

“下一步”指的是某个功能执行中的“下一步”,当然也可能是编程语境下的“ 下一步 ”,即下一条语句, 所以这里描述的是异步操作,如保存文件操作,服务器等待连接、网络下载等。

举个栗子, 创建了QTcpServer对象并调用listen() 函数监听连接,这时如果调用QTcpServer::waitForNewConnection(...)函数,就会阻塞程序的运行,直到连接到来函数才能返回,进而执行下一句。这里,listen()函数“ 触发 ”了监听行为, “ 下一步 ”与网络连接相关的动作要等连接到来后才能执行,而 QTcpServer::waitForNewConnection(...)则作为我们是否执行下一步的判断标准,只不过这里用的是阻塞的方式;对于异步操作,也可以用非阻塞的方式解决,Qt的信号和槽机制就能很好的解决,如我们可以接收newConnection()信号判断连接是否到来,而不必将程序阻塞在那。

保持Qt GUI响应的几种方法的更多相关文章

  1. Qt中显示图像的两种方法

    博客转载自:https://blog.csdn.net/lg1259156776/article/details/52325361 在Qt中处理图片一般都要用到QImage类,但是QImage的对象不 ...

  2. 解决MySQL安装到最后一步未响应的三种方法

    这种情况一般是你以前安装过MySQL数据库服务项被占用了.解决方法: 方法一:安装MySQL的时候在这一步时它默认的服务名是“MySQL” 只需要把这个名字改了就可以了.可以把默认的服务器的名称手动改 ...

  3. Qt 自定义事件(三种方法:继承QEvent,然后Send Post就都可以了,也可以覆盖customEvent函数,也可覆盖event()函数)

    Qt 自定义事件很简单,同其它类库的使用很相似,都是要继承一个类进行扩展.在 Qt 中,你需要继承的类是 QEvent. 继承QEvent类,你需要提供一个QEvent::Type类型的参数,作为自定 ...

  4. Qt 设置背景图片3种方法(三种方法:QPalette调色板,paintEvent,QSS)

    方法1. setStylSheet{"QDialog{background-image:url()"}}  //使用styleSheet 这种方法的好处是继承它的dialog都会自 ...

  5. Jmeter查看结果树之查看响应的13种方法[详解]

    查看结果树查看响应有哪几种方法,可通过左侧面板底部的下拉框选择 1.Text 查看结果树中请求的默认格式为text,会显示请求的取样器结果.请求.响应数据3个部分内容. 取样器结果: 默认Raw视图, ...

  6. GNU Radio下QT GUI Tab Widget的使用方法

    期望显示出的效果: 即将要显示的图放在各自的标签页中. 整体框图: 具体设置: QT GUI Tab Widget的设置: 其中 ID改为自己想改的,这里我写的是display GUI Hint所代表 ...

  7. 【Qt开发】Qt中显示图像的两种方法对比

    在Qt中处理图片一般都要用到QImage类,但是QImage的对象不能够直接显示出来,要想能看到图片,初步发现有两种方法. 一.QImage转QPixmap,然后用QLabel::setPixmap( ...

  8. jmeter 查看结果树之查看响应的13种方法

    查看结果树查看响应有哪几种方法,可通过左侧面板底部的下拉框选择 1 Text 查看结果树中请求的默认格式为Text,显示取样器结果.请求.响应数据3个部分内容. 取样器结果: 默认Raw展示,可以切换 ...

  9. Qt产生随机数(两种方法)

    第一种方法 #include <QTime> QTime time; time= QTime::currentTime(); qsrand(time.msec()+time.second( ...

随机推荐

  1. iOS开源项目周报1215

    由OpenDigg 出品的iOS开源项目周报第一期来啦.我们的iOS开源周报集合了OpenDigg一周来新收录的优质的iOS开发方面的开源项目,方便iOS开发人员便捷的找到自己需要的项目工具等. PY ...

  2. App解读

    一直不懂别人口中说的原生开发.混合式开发.今天突然看了一篇文章讲解的是什么叫做原生App?移动 Web App?混合APP?分享给大家. 原生App是专门针对某一类移动设备而生的,它们都是直接安装到设 ...

  3. python 常用第三方模块

    除了内建的模块外,Python还有大量的第三方模块. 基本上,所有的第三方模块都会在https://pypi.python.org/pypi上注册,只要找到对应的模块名字,即可用pip安装. 本章介绍 ...

  4. 使用T4模板生成不同部署环境下的配置文件

    在开发企业级应用的时候,通常会有不同的开发环境,比如有开发环境,测试环境,正式环境,生产环境等.在一份代码部署到不同环境的时候,不同环境的配置文件可能需要根据目标环境不同而不同.比如在开发环境中,数据 ...

  5. MapReduce剖析笔记之五:Map与Reduce任务分配过程

    在上一节分析了TaskTracker和JobTracker之间通过周期的心跳消息获取任务分配结果的过程.中间留了一个问题,就是任务到底是怎么分配的.任务的分配自然是由JobTracker做出来的,具体 ...

  6. iOS 之消息推送(个推)---个人小结

    前言:自从上个星期开始整这个推送,弄了差不多一个星期,今天终于给整好了,因此现在来记录这段"奇妙"的旅程. 我们公司使用的消息推送是用的第三方--个推,这里不得不说一下,个推的技术 ...

  7. wordpress去掉导航栏链接中的category

    找到服务器目录下的functions..php文件,在末尾处添加如下内容即可. 路径:/htdocs/wp-content/themes/functions.php 要追加在functions.php ...

  8. 关于项目中下单流程HTML设计的一些思考

    需求 上面文字和圈圈是对齐的. 想法 一开始是想把文字和圈圈分开来的,也就是两个盒子放置.但操作中发现,想把它们对齐非常的难,总有一些是无法对齐的. 最终换了一种实现方式,按照需求,不就是想把它们关联 ...

  9. 04.移动先行之谁主沉浮----XAML的探索

    如果移动方向有任何问题请参考===> 异常处理汇总-移动系列(点) 移动先行之谁主沉浮? 带着你的Net飞奔吧! 链接======>(点) XMAL引入 XAML 类似于 HTML,是一种 ...

  10. 2.Kali安装VMware tools(详细+异常处理)

    dnt@MT:~$ cd /media/cdrom0 进入光驱内 dnt@MT:/media/cdrom0$ ls 查看当前目录下有哪些内容manifest.txt run_upgrader.sh V ...