如何为无法获取进度百分比的耗时操作增加“伪进度条”?

在实际开发中,可能会遇到这样的场景:“一个操作非常耗时,但却无法获取其进度百分比”。造成这种情况的原因可能有:
1)该操作属于第三方库(泛指我们使用但无法修改的第三方代码,因此质量有高有底),可能由于第三方库作者没有意识到这个操作在某些情况下会非常耗时,没有提供进度值。
2)某些操作压根就无法计算进度或者计算进度要牺牲极大的效率。

这种场景,执行这样的耗时操作会导致程序UI假死。为了提高用户体验,我们可以为该操作提供一个假的进度条,即保证程序UI不假死。我们将耗时操作放到子线程中去执行,然后在GUI主线程中抛出一个进度对话框,用一个定时器每隔0.5秒(或者一个更合适的值)更新进度条的百分比。用一个循环待子线程执行完毕则GUI主线程继续往下执行。

void MainWindow::slotRun()
{
    QProgressDialog progressDlg(tr("正在读取文件..."), tr("取消"),
                                0, 100, this,
                                Qt::CustomizeWindowHint);
    progressDlg.setWindowModality(Qt::WindowModal);
    progressDlg.setMinimumSize(400, 100);
    progressDlg.show(); // 进度对话框在GUI主线程中执行

    FileReader aFileReader("D:/BigFile.txt");
    FunctionRunThread runThread(aFileReader);  // 将耗时操作放到子线程中去执行
    runThread.start();
    int cnt = 0; // 循环次数计数器,用于"计算"当前的进度值
    while (!runThread.isFinished())  // 只要子线程还没有完成,就一直循环,并更新进度条
    {
        //QThread::currentThread()->wait(500);
        //QApplication::instance()->thread()->wait(500);

        progressDlg.setValue((cnt++%20)*5); // 进度值每次递增5%,达到100后则再次从0开始

        QEventLoop eventloop; // 使用事件循环阻塞主线程
        QTimer::singleShot(500, &eventloop, SLOT(quit())); // wait 0.5s
        eventloop.exec(); // 每0.5秒执行一次事件循环,然后更新进度条
    }
    progressDlg.setValue(100);
    qDebug() << "cnt=" << cnt;
}

偶然看到一个更高级的实现方式:
http://wiki.qt.io/Progress_Bar

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值