Playing with coroutines and Qt
你好!
我最近想知道C ++中的协程的状态,我发现了几个实现。我决定选择一个用于我的实验。
它简单易用,适用于Linux和Windows。
我的目标是试图找到一种方法来让代码异步运行,而不必等待信号触发插槽,并避免调用QCoreApplication :: processEvents或在堆栈中创建QEventLoop。
我的第一种方法是将自定义事件调度程序的processEvent函数转换为协程并使用yield。几次失败后,我决定不继续这种方式。
我的下一个尝试是将一个插槽转换为协程:
|
1
|
QTimer::singleShot(0, std::bind(&coroutine::resume, coroutine::create([]() { ... }); |
在这个lambda中,CPU将执行代码直到yield,它将跳回到应用程序事件循环。
完整的代码是:
|
1
2
3
4
五
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
|
#include "coroutine.h"#include <QtCore>#include <QtWidgets>int main(int argc, char **argv){ QApplication app(argc, argv); QPushButton fibonacciButton("0: 0"); fibonacciButton.show(); QObject::connect(&fibonacciButton, &QPushButton::pressed, std::bind(&coroutine::resume, coroutine::create([&]() { qulonglong f0 = 1, f1 = 0, n = 1; fibonacciButton.setText(QString("1: 1")); coroutine::yield(); fibonacciButton.setText(QString("2: 1")); forever { auto next = f1 + f0; f0 = f1; f1 = next; fibonacciButton.setText(QString("%0: %1").arg(n++).arg(f0 + f1)); coroutine::yield(); } }))); return app.exec();} |
在这里我们可以看到一个连接到一个lambda函数的按钮,它可以计算斐波那契数列中的数字。在计算下一个数字之后,我们调用yield,它将从该函数跳转到事件循环。当用户再次按下该按钮时,代码将在收益率之后返回到下一行。
这个例子的作用是因为用户需要再次按下按钮来恢复代码的执行。
但是,有时我们想要自动恢复执行。为此,我们需要产生执行并安排执行的简历:
|
1
2
3
4
五
6
|
void qYield(){ const auto routine = coroutine::current(); QTimer::singleShot(0, std::bind(&coroutine::resume, routine)); coroutine::yield();} |
第一行获取协程的标识符,第二行安排恢复。有了yield,CPU会回到以前的帧,最后到达主循环,并且无条件恢复代码。
下一步是在情况发生时尝试恢复。Qt提供了指示什么时候发生的信号,所以更优化的执行方式是:
|
1
2
3
4
五
6
7
8
9
|
template <typename Func>void qAwait(const typename QtPrivate::FunctionPointer::Object *sender, Func signal){ const auto routine = coroutine::current(); const auto connection = QObject::connect(sender, signal, std::bind(&coroutine::resume, routine)); coroutine::yield(); QObject::disconnect(connection);} |
我们创建一个临时连接来恢复插槽的执行,而不是排队恢复。
一个例子可以是:
|
1
2
3
4
五
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
#include "coroutine.h"#include <QtCore>#include <QtWidgets>#include <QtNetwork>int main(int argc, char **argv){ QApplication app(argc, argv); QPlainTextEdit textEdit; textEdit.show(); QTimer::singleShot(0, std::bind(&coroutine::resume, coroutine::create([&]() { QUrl url("http://download.qt.io/online/qt5/linux/x64/online_repository/Updates.xml"); QNetworkRequest request(url); QNetworkAccessManager manager; auto reply = manager.get(request); qAwait(reply, &QNetworkReply::finished); textEdit.setPlainText(reply->readAll()); reply->deleteLater(); }))); return app.exec();} |
在这里,我创建了一个QTextEdit,它从互联网上接收文件的内容。当QNetworkReply完成时,数据被写入QTextEdit。
另一个例子:
|
1
2
3
4
五
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
|
#include "coroutine.h"#include <QtCore>#include <QtWidgets>#include <QtNetwork>int main(int argc, char **argv){ QApplication app(argc, argv); QPlainTextEdit edit; edit.show(); QTimer::singleShot(0, std::bind(&coroutine::resume, coroutine::create([&]() { auto previousText = edit.toPlainText(); forever { if (edit.toPlainText() == QStringLiteral("quit")) { qApp->quit(); } else if (previousText != edit.toPlainText()) { qDebug() << previousText << "->" << edit.toPlainText(); previousText = edit.toPlainText(); } qAwait(&edit, &QPlainTextEdit::textChanged); qDebug() << "QPlainTextEdit::textChanged"; } }))); return app.exec();} |
每次用户修改文本时,该应用程序都会打印文本,并在用户写入“退出”字时完成执行。
https://blog.qt.io/blog/2018/05/29/playing-coroutines-qt/
Playing with coroutines and Qt的更多相关文章
- Qt: The State Machine Framework 学习
State Machine,即为状态机,是Qt中一项非常好的框架.State Machine包括State以及State间的Transition,构成状态和状态转移.通过状态机,我们可以很方便地实现很 ...
- qt Multimedia 模块类如何使用?
qt 多媒体模块介绍 类名 英文描述 中文描述 QAudioBuffer Represents a collection of audio samples with a specific format ...
- Qt Multimedia 模块类如何使用?(表格)
qt 多媒体模块介绍 类名 英文描述 中文描述 QAudioBuffer Represents a collection of audio samples with a specific format ...
- API Design Principles -- QT Project
[the original link] One of Qt’s most reputed merits is its consistent, easy-to-learn, powerfulAPI. T ...
- qtchooser - a wrapper used to select between Qt development binary(2种方法)
---------------------------------------------------------------------------------------------------- ...
- GStreamer基础教程11 - 与QT集成
摘要 通常我们的播放引擎需要和GUI进行集成,在使用GStreamer时,GStreamre会负责媒体的播放及控制,GUI会负责处理用户的交互操作以及创建显示的窗口.本例中我们将结合QT介绍如何指定G ...
- 简单的多屏播放器示例(vlc+qt)
介绍 简单的多屏播放器 最多同时播放16个视频 支持本地文件和rtsp.rtmp等流媒体播放 VS2015工程,依赖Qt+VLC 练手作品 截图 下载 程序:download.csdn.net/d ...
- vlc音视频开发(一)环境搭建(qt篇)
来源:微信公众号「编程学习基地」 目录 简介 qt配置vlc环境 simple_libvlc_qt_player 项目地址 简介 VLC 是一款自由.开源的跨平台多媒体播放器及框架,可播放大多数多媒体 ...
- QT内省机制、自定义Model、数据库
本文将介绍自定义Model过程中数据库数据源的获取方法,我使用过以下三种方式获取数据库数据源: 创建 存储对应数据库所有字段的 结构体,将结构体置于容器中返回,然后根据索引值(QModelIndex) ...
随机推荐
- 带你彻彻底底弄懂Scroller
Scroller的使用 这是一个滑动帮助类.并不能够使View真正的滑动,而是依据时间的流逝.获取插值器中的数据.传递给我们.让我们去配合scrollTo/scrollBy去让view产生缓慢滑动,产 ...
- android图像处理系列之五-- 给图片添加边框(中)
前面一篇讲到给图片加边框的方式,只能给图片加一些有规则的边框,如果想加一些比较精美的效果,就有点麻烦了.下面就给出解决这个问题的思路. 思路是:一些比较精美的花边图片我们是很难用代码控制,就目前本人水 ...
- RTSP、HTTP、HTTPS、SDP四种协议详解
我们将主要讲解RTSP,HTTP,HTTPS, SDP四种协议. 一:RTSP协议简介 实时流协议RTSP是一个应用层协议,用于控制具有实时特性的数据(例如多媒体流)的传送. RTSP协议一般与RT ...
- 深入理解Android(3)——Eclipse集成javah和NDK-Builder
在上一篇文章中我们使用了javah工具来生成了native java文件所对应的C++头文件,但是这样生成比较麻烦,我们这一篇来介绍如何在eclipse中集成javah和NDK-Builder. 一. ...
- Android java.lang.NoSuchFieldError: No static field xxx of type I in class Lcom/XX/R$id; or its superclasses
项目开发快到尾声,突然发现之前一个模块莫名其妙的奔溃了,我的内心也是奔溃的.以前一直都是好好的,也没去动过它,为啥会出现这样的问题呢? 下面我会根据自己的理解来看待问题 android是怎么根据id查 ...
- 【bzoj4864】神秘物质
Description 给出一个长度为n的序列,第i个数为ai,进行以下四种操作: merge x e:将当前第x个数和第x+1个数合并,得到一个新的数e: insert x e:在当前第x个数和第x ...
- shell项目-分发系统-构建文件分发系统
shell项目-分发系统-构建文件分发系统 需求背景对于大公司而言,肯定时不时会有网站或者配置文件更新,而且使用的机器肯定也是好多台,少则几台,多则几十甚至上百台.所以,自动同步文件是至关重要的. 实 ...
- javaEE之------Spring-----》 AspectJ注解
前面介绍了下Spring中的切面技术.如今说下採用注解的方式进行切面 首先肯定和之前的一样.须要一个自己主动代理的注解类 AnnotationAwareAspectJAutoProxyCreator ...
- TextView -无法调节字体、边框的距离
今天调节一个字体边框距离,结果一直都实现不了,布局如下 <RelativeLayout xmlns:android="http://schemas.android.com/apk/re ...
- Android广告轮播图实现
先看效果 第一步,布局 <?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmln ...