简述

QFuture 表示异步计算的结果,QFutureWatcher 则允许使用信号和槽监视 QFuture,也就是说,QFutureWatcher 是为 QFuture 而生的。

  • 简述
  • 详细描述
  • 基本使用
  • 更多参考

详细描述

QFutureWatcher 提供了有关 QFuture 的信息和通知,使用 setFuture() 函数开始监视一个特定的 QFuture,函数 future() 则返回由 setFuture() 设置的 future。

为了方便,QFuture 的很多函数可以直接通过 QFutureWatcher 来访问,例如:progressValue()、progressMinimum()、progressMaximum()、progressText()、isStarted()、isFinished()、isRunning()、isCanceled()、isPaused()、waitForFinished()、result() 和 resultAt()。而 cancel()、setPaused()、pause()、resume() 和 togglePaused() 是 QFutureWatcher 中的槽函数。

状态更改由 started()、finished()、cancelled()、paused()、resumed()、resultReadyAt() 和 resultsReadyAt() 信号提供,进度信息由 progressRangeChanged()、progressValueChanged() 和progressTextChanged() 信号提供。

由函数 setPendingResultsLimit() 提供节流控制。当挂起的 resultReadyAt() 或 resultsReadyAt() 信号数量超过限制时,由 future 表示的计算将被自动节流。一旦挂起的信号数量下降到限制以下时,计算将恢复。

示例,开始计算并当完成时获取槽回调:

// 实例化对象,并连接到 finished() 信号。
MyClass myObject;
QFutureWatcher<int> watcher;
connect(&watcher, SIGNAL(finished()), &myObject, SLOT(handleFinished()));// 开始计算
QFuture<int> future = QtConcurrent::run(...);
watcher.setFuture(future);

基本使用

来看一个图像加载和缩放的示例。选择多个图片,进行异步计算(将所有图片进行缩放),加载过程中可以显示进度,以便我们实时了解进展。每当一个图片处理完成,就会显示在窗体中。

这里仅为了演示效果,加载了 8 张 图片。

具体的源码如下所示:

ImagesView.h:

#ifndef IMAGES_VIEW_H
#define IMAGES_VIEW_H#include <QFutureWatcher>
#include <QWidget>class QLabel;
class QPushButton;
class QVBoxLayout;
class QGridLayout;class ImagesView : public QWidget
{Q_OBJECTpublic:explicit ImagesView(QWidget *parent = 0);~ImagesView();private slots:void open();  // 打开目录,加载图片void showImage(int index);  // 显示图片void finished();  // 更新按钮状态private:QPushButton *m_pOpenButton;QPushButton *m_pCancelButton;QPushButton *m_pPauseButton;QVBoxLayout *m_pMainLayout;QGridLayout *m_pImagesLayout;QList<QLabel *> labels;QFutureWatcher<QImage> *m_pWatcher;
};#endif // IMAGES_VIEW_H

下面是实现部分,c_nImageSize 表示的是图片被缩放的大小(宽度:100 px,高度:100px),函数 scale() 则是对图片缩放的具体实现。

ImagesView.cpp

#include <QLabel>
#include <QPushButton>
#include <QProgressBar>
#include <QFileDialog>
#include <QtConcurrent/QtConcurrentMap>
#include <QStandardPaths>
#include <QHBoxLayout>
#include <qmath.h>
#include "ImagesView.h"const int c_nImageSize = 100;// 缩放图片
QImage scale(const QString &imageFileName)
{QImage image(imageFileName);return image.scaled(QSize(c_nImageSize, c_nImageSize), Qt::IgnoreAspectRatio, Qt::SmoothTransformation);
}ImagesView::ImagesView(QWidget *parent): QWidget(parent)
{setWindowIcon(QIcon(":/Images/logo"));setWindowTitle(QStringLiteral("Qt之QFutureWatcher"));resize(800, 600);// 初始化控件m_pWatcher = new QFutureWatcher<QImage>(this);m_pOpenButton = new QPushButton(QStringLiteral("打开图片"));m_pCancelButton = new QPushButton(QStringLiteral("取消"));m_pPauseButton = new QPushButton(QStringLiteral("暂停/恢复"));QProgressBar *pProgressBar = new QProgressBar(this);m_pCancelButton->setEnabled(false);m_pPauseButton->setEnabled(false);// 布局QHBoxLayout *pButtonLayout = new QHBoxLayout();pButtonLayout->addWidget(m_pOpenButton);pButtonLayout->addWidget(m_pCancelButton);pButtonLayout->addWidget(m_pPauseButton);pButtonLayout->addStretch();pButtonLayout->setSpacing(10);pButtonLayout->setMargin(0);m_pImagesLayout = new QGridLayout();m_pMainLayout = new QVBoxLayout();m_pMainLayout->addLayout(pButtonLayout);m_pMainLayout->addWidget(pProgressBar);m_pMainLayout->addLayout(m_pImagesLayout);m_pMainLayout->addStretch();m_pMainLayout->setSpacing(10);m_pMainLayout->setContentsMargins(10, 10, 10, 10);setLayout(m_pMainLayout);// 连接信号槽 - 加载、显示进度、打开、取消等操作connect(m_pWatcher, SIGNAL(resultReadyAt(int)), SLOT(showImage(int)));connect(m_pWatcher, SIGNAL(progressRangeChanged(int,int)), pProgressBar, SLOT(setRange(int,int)));connect(m_pWatcher, SIGNAL(progressValueChanged(int)), pProgressBar, SLOT(setValue(int)));connect(m_pWatcher, SIGNAL(finished()), SLOT(finished()));connect(m_pOpenButton, SIGNAL(clicked()), SLOT(open()));connect(m_pCancelButton, SIGNAL(clicked()), m_pWatcher, SLOT(cancel()));connect(m_pPauseButton, SIGNAL(clicked()), m_pWatcher, SLOT(togglePaused()));
}ImagesView::~ImagesView()
{m_pWatcher->cancel();m_pWatcher->waitForFinished();
}// 打开目录,加载图片
void ImagesView::open()
{// 如果已经加载图片,取消并进行等待if (m_pWatcher->isRunning()) {m_pWatcher->cancel();m_pWatcher->waitForFinished();}// 显示一个文件打开对话框QStringList files = QFileDialog::getOpenFileNames(this,QStringLiteral("选择图片"),QStandardPaths::writableLocation(QStandardPaths::PicturesLocation),"*.jpg *.png");if (files.count() == 0)return;// 做一个简单的布局qDeleteAll(labels);labels.clear();int dim = qSqrt(qreal(files.count())) + 1;for (int i = 0; i < dim; ++i) {for (int j = 0; j < dim; ++j) {QLabel *pLabel = new QLabel(this);pLabel->setFixedSize(c_nImageSize, c_nImageSize);m_pImagesLayout->addWidget(pLabel, i, j);labels.append(pLabel);}}// 使用 mapped 来为 files 运行线程安全的 scale 函数m_pWatcher->setFuture(QtConcurrent::mapped(files, scale));m_pOpenButton->setEnabled(false);m_pCancelButton->setEnabled(true);m_pPauseButton->setEnabled(true);
}// 显示图片
void ImagesView::showImage(int index)
{labels[index]->setPixmap(QPixmap::fromImage(m_pWatcher->resultAt(index)));
}// 更新按钮状态
void ImagesView::finished()
{m_pOpenButton->setEnabled(true);m_pCancelButton->setEnabled(false);m_pPauseButton->setEnabled(false);
}

构造函数中,需要注意的是槽函数,其中 resultReadyAt() 表示 index 对应位置的处理结果已准备就绪,所以连接该信号至槽函数 showImage(),可以显示处理完的图片。

为了显示处理进度,我们构造了一个进度条,当 QFutureWatcher 的 progressRangeChanged() 的信号发射时,进度条的范围会发生改变,而 progressValueChanged() 信号发射时,会更新进度条的值。

如果加载的图片较多时,可以通过点击“取消”按钮,这时会调用 QFutureWatcher 的 cancel() 槽函数来取消计算。“暂停/恢复”则调用 togglePaused() 槽函数,用于切换异步计算的暂停状态,换句话说,如果计算当前已暂停,调用此函数将进行恢复;如果计算正在运行,则会暂停。

当点击“打开”按钮时,会调用槽函数 open(),默认打开图片目录,以便进行图片的选择。然后根据图片创建对应数量的标签 QLabel,用于显示后期缩放的图片。创建完成后,使用 mapped() 进行并行计算,并添加至 QFutureWatcher 中,让其使用信号和槽监视 QFuture。

接下来,就可以直接使用了。

#include <QApplication>
#include "ImagesView.h"int main(int argc, char *argv[])
{QApplication app(argc,argv);ImagesView view;view.show();return app.exec();
}

这样,我们就完成了一个图片缩放加载缩放的功能。

更多参考

  • Qt之Concurrent框架
  • Qt之Concurrent Map和Map-Reduce
  • Qt之Concurrent Filter和Filter-Reduce
  • Qt之Concurrent Run
  • Qt之QFuture

Qt之QFutureWatcher相关推荐

  1. Qt之Concurrent框架

    简述 QtConcurrent 命名空间提供了高级 API,使得可以在不使用低级线程原语(例如:互斥.读写锁.等待条件或信号量)的情况下编写多线程程序,使用 QtConcurrent 编写的程序根据可 ...

  2. photo mosaic 拼图马赛克

    photo mosaic 好多张图片拼成一张完整的图片 原理: 读取原始图片,进行分割,分析每个小块RGB均值,并存储. 读取若干图片,提取并存储各个图片的RGB均值.与原始图片的每个小块进行匹配. ...

  3. 【Qt】Qt再学习(九):并发 QtConcurrent、QFuture、QFutureWatcher

    1.QtConcurrent 该QtConcurrent命名空间提供高层次的API,使人们有可能不写使用低级线程原语的多线程程序,如互斥,读写锁,等待条件或信号.用QtConcurrent编写的程序会 ...

  4. 【转】【QT】 Threads, Events and QObjects

    前言: qt wiki 中这篇文章3月份再次更新,文章对 QThread 的用法,使用场景,有很好的论述,可以作为 Qt 多线程编程的使用指南,原文在这里,原作者 peppe 开的讨论贴在这里. 原文 ...

  5. Qt 并行计算圆周率示例

    Qt 并行计算圆周率示例 简介: 因为最近的一项项目中要用到并行计算,所以花了两天的时间了解了下Qt的并行计算的功能,顺便也尝试写了一个Demo和大家一起分享. 任务如下: 1.实现多种方法计算圆周率 ...

  6. Qt 并行计算 Concurrent Run的翻译

    资料来源:https://doc.qt.io/qt-5/qtconcurrentrun.html Concurrent Run的简介: QtConcurrent::run() 是开启单独一个线程来运行 ...

  7. qt qthead里如何响应信号_Qt 中的多线程技术(翻译)

    原文链接 Multithreading Technologies in Qt​doc.qt.io 正文 Qt 提供一系列的类与函数来处理多线程.Qt 开发者们可以使用下面四种方法来实现多线程应用. Q ...

  8. Qt多线程编程的主要线程类

    1.描述 Qt提供QThread类处理多线程,继承自QObject.不受平台影响,实现跨平台功能. 2.主要的线程类 QAtomicInt:提供Interger与平台无关的Atomic运算,即提供了整 ...

  9. Qt文档阅读笔记-QtConcurrent Progress Dialog Example解析

    这篇展示了如何监听任务的进度. QtConcurrent Progress Dialog使用QFutrueWathcer类去监听任务进程进展. 代码如下: progressdialog.pro QT ...

最新文章

  1. 数据库高性能读写分离集群操作说明
  2. Java报表工具FineReport导出EXCEL的四种API
  3. BAT架构师技术文档:Redis+Nginx+Dubbo精选+面试题+架构师精选视频(免费领)
  4. iOS单例创建的一点疑惑
  5. Doctype? 严格模式与混杂模式-如何触发这两种模式,区分它们有何意义?
  6. Java Mail+Thymeleaf模板引擎实现发送HTML格式邮件
  7. 如何计算两个日期之间相差的天数?
  8. 395. Longest Substring with At Least K Repeating Characters
  9. 通达信板块监控指标_通达信洞察强势板块指标公式
  10. 再战“超融合”,戴尔、Nutanix绝世好CP
  11. CentOS上使用Docker安装Redis-Cluster (redis6.x)
  12. 携程apollo系列-个人开发环境搭建
  13. 国内 RISC-V 产学研基地成立,Intel、Arm、RISC-V 将三分天下?
  14. jenkins简介及docker部署
  15. 关于readonly修饰符
  16. Datalogic得利捷推出物流应用领域全新标杆产品——AV900
  17. PHP 大小写转换函数 lcfirst ucfirst ucwords strtolower strtoupper
  18. 推荐系统(十一)阿里深度兴趣网络(一):DIN模型(Deep Interest Network)
  19. odi12配置mysql_Oracle数据库之Oracle ODI 12c之多表联合查询以及定时任务设置
  20. 读jQuery 权威指南[5]-插件

热门文章

  1. 为JS和C#类加一个扩展方法吧:P
  2. 《Redis设计与实现》之第七章:压缩列表
  3. PHPExcel对于Excel中日期和时间类型的处理
  4. [LUOGU] P2330 [SCOI2005]繁忙的都市
  5. Dockerfile 最佳实践
  6. JavaScript:引用js文件时的编码格式问题
  7. 模板方法模式(Template Pattern)
  8. SVN:服务器资源删掉,本地添加时和删掉的名字同名出现One or more files are in a conflicted state....
  9. go语言中无法获取goroutine相关的信息
  10. adb 连接不上电脑怎么办?