Qt 如何处理密集型耗时的事情(频繁调用QApplication::processEvents)

  有时候需要处理一些跟界面无关的但非常耗时的事情,这些事情跟界面在同一个线程中,由于时间太长,导致界面无法响应,处于“假死”状态。例如:在应用程序中保存文件到硬盘上,从开始保存直到文件保存完毕,程序不响应用户的任何操作,窗口也不会重新绘制,从而处于“无法响应”状态,这是一个非常糟糕的体验 。

     在这种情况下,有一种方法是使用多线程,即在子线程中处理文件保存,主线程负责界面相关。

     而如果不想使用多线程,最简单的办法就是在文件保存过程中频繁调用QApplication::processEvents()。该函数的作用是让程序处理那些还没有处理的事件,然后再把使用权返回给调用者。

代码如下:

 1 bool MyApp::writeFile(const QString &filename)
 2 {
 3      QFile file(filename);
 4 ...
 5     QApplication::setOverrideCursor(Qt::WaitCursor);
 6      for(int r = 0; r != rowCount; ++r)
 7      {
 8           for(int c = 0; c != colCount; ++c)
 9           {
10                out << table(r,c);   
11                qApp.processEvents();
12           }
13      }
14     QApplication::restoreOverrideCursor();
15 }

这样一来,程序就能响应了。

     但是,该方法有一个问题:可能正在保存文件的过程中,用户不小心又单击了保存,或不小心关闭了程序主窗口,这样会产生意想不到的后果。

     解决这个问题的最简单的办法是替换成:

qApp->processEvents(QEventLoop::ExcludeUserInputEvents);//它可以忽略用户的输入(鼠标和键盘事件)。

 进一步的,如果想显示一个带有进度条的对话框,随时显示当前的进度状态,可以使用QProgressDialog。

 1 bool MyApp::writeFile(const QString &filename)
 2 {
 3      QFile file(filename);
 4 ...
 5     QApplication::setOverrideCursor(Qt::WaitCursor);
 6      QProgressDialog progress;
 7      progress.setWindowTitle(tableData->sNameCH);
 8      progress.setLabelText(QStringLiteral("数据保存中,请稍候..."));
 9      //progress.setCancelButton(0);//不显示“取消”按钮
10      progress.setCancelButtonText("取消");
11      progress.setRange(0,rowCount );
12      progress.setModal(true);
13      //此处没有调用show()来显示,是因为QProgressDialog会自动决定是否显示
14      //如果时间过短,就不会显示。
15      for(int r = 0; r != rowCount; ++r)
16      {
17           progress.setValue(row);
18           //如果用户单击了“取消”,就取消保存文件,并删除该文件。
19           if(progress.wasCanceled)
20           {
21                file.remov();
22                return false;
23           }
24           for(int c = 0; c != colCount; ++c)
25           {
26                out << table(r,c);   
27                qApp.processEvents();
28           }
29      }
30     QApplication::restoreOverrideCursor();
31 }

显示效果如下:

ScreenClip

原文地址:https://www.cnblogs.com/ybqjymy/p/13202865.html