Qt 如何处理密集型耗时的事情

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

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

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

代码以下:oop

bool MyApp::writeFile(const QString &filename)
{
     QFile file(filename);
...
    QApplication::setOverrideCursor(Qt::WaitCursor);
     for(int r = 0; r != rowCount; ++r)
     {
          for(int c = 0; c != colCount; ++c)
          {
               out << table(r,c);   
               qApp.processEvents();
          }
     }
    QApplication::restoreOverrideCursor();
}

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

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

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

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

 

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

bool MyApp::writeFile(const QString &filename)
{
     QFile file(filename);
...
    QApplication::setOverrideCursor(Qt::WaitCursor);
     QProgressDialog progress;
     progress.setWindowTitle(tableData->sNameCH);
     progress.setLabelText(QStringLiteral("数据保存中,请稍候..."));
     //progress.setCancelButton(0);//不显示“取消”按钮
     progress.setCancelButtonText("取消");
     progress.setRange(0,rowCount );
     progress.setModal(true);
     //此处没有调用show()来显示,是由于QProgressDialog会自动决定是否显示
     //若是时间太短,就不会显示。
     for(int r = 0; r != rowCount; ++r)
     {
          progress.setValue(row);
          //若是用户单击了“取消”,就取消保存文件,并删除该文件。
          if(progress.wasCanceled)
          {
               file.remov();
               return false;
          }
          for(int c = 0; c != colCount; ++c)
          {
               out << table(r,c);   
               qApp.processEvents();
          }
     }
    QApplication::restoreOverrideCursor();
}

显示效果以下:blog

ScreenClip

相关文章
相关标签/搜索