QML 下的多线程



  • 说到 QML 的多线程,一般就会提起 WorkerScript。由于 JavaScript 可以方便的被隔离上下文(QML的上下文),所以 JavaScript 就可以被作为线程运行的代码。但是 WorkerScript 使用起来最大的缺点,就是只能使用 JavaScript,容不得半丁的 Qt 对象的染指。

    所以我尝试了一个新的项目,将一个 QJSEngine 移动到 QThread ,然后通过自定的插件注册方式,将c++强大的运算能力迁移到了多线程中的 JavaScript

    项目在这里QmlThread

    不过,我发现 QJSEngine 本身功能就太少了,连 consoleXmlHttpRequest 对象都没有。

    所以我决定使用 QML 作为多线程的脚本,不过由于 QML 语言本身的问题,可能会毫无预兆的读写到主线程中,QQmlEngine 的上下文。所以这是一个潘多拉的魔盒啊。

    现在给出一些代码示例:

    下图是跑在线程的 Qml 文件

    0_1466173588001_upload-7d2a7f42-9807-4120-8ef5-e234f0d52866

    下图是主线程中调用的地方:

    0_1466173615216_upload-c7ee7236-f002-4340-af26-85529c77cc15

    接口设计可能参照 WebScoket 那样,就像主线程异步连接到了网络中一台功能强劲的电脑,将你的请求转化为运算,然后再将结果发送回主线程。



  • @qyvlik 我们继续深入地讨论一下:
    Qt 5.7呢,可以调试多个QJSEngine了。
    然后呢,我这里看来可以将核心多线程,每个线程一个QJSEngine的代码放在C++部分,然后和以前一样写QML代码。



  • @jiangcaiyang123

    其实我想向 Qt 提个特性。就是 QML 的多线程。

    例如文件命名为 xxx.thread.qml 的文件不仅在编辑器上支持代码高亮和补全,而且还可以检测到模块引用,主线程上下文引用。

    xx.ui.qml 一样的性质。



  • @qyvlik 这个可能要涉及到不同的QQmlEngine中QML和C++的相互操作,以及多线程的调度,还有就是全局单例的访问问题了。只要有恒心还是可以实现的。



  • @jiangcaiyang123

    只要能完全检测出 xxx.thread.qml 使用的模块和是否使用到了主线程的上下文,就行了。

    这里说道主线程上下文,一般指,入口qml文件的属性和id,可以被qml路径树中其他qml对象访问。

    我们必须禁止直接在QML层面共享上下文,共享上下文应该在c++中处理。



  • good job!



  • @jiangcaiyang123 我仔细想了想,QML 中的单例是依赖于一个 QQmlEngine 的。所以单例完全没有问题。



  • @qyvlikQML 下的多线程 中说:

    @jiangcaiyang123

    只要能完全检测出 xxx.thread.qml 使用的模块和是否使用到了主线程的上下文,就行了。

    这里说道主线程上下文,一般指,入口qml文件的属性和id,可以被qml路径树中其他qml对象访问。

    我们必须禁止直接在QML层面共享上下文,共享上下文应该在c++中处理。

    这个只能通过编程提示的方式来保证了,无法禁止。

    还有做到线程QML文件是一个单例,是没有问题的,不过单例要表现得像QML单例,即QQmlEngine可被多个线程共享,我就觉得有些难了。



  • @jiangcaiyang123

    QML 单例是依赖于 QQmlEngine 的,如果要c++那种,整个程序以及程序的线程都可以共享的话,这个单例对象在使用和维护上开销不小。

    所以才会有 thread_local 这个线程存储对象。

    QML 单例更多是某个 QQmlEngine 上下文对象。



  • @qyvlik

    QML 中的多线程,一般是使用 QJsonValue 和有线程保护的 QObject* 作为信使。

    怎么设计出像 ListModel 那样支持投递线程中进行异步加载资源的数据模型呢?



  • 好想法,在邮件列表里提一下怎么样?



  • @qyvlik 你想要什么样的QML多线程效果了。



  • @jiangcaiyang 一个线程放一个 QQmlEngine,然后使用线程池来处理多个 QQmlEngine。然后还可以在线程中安全的处理 QAbstractListModel

    大体功能如下:

    1. 一个线程一个 QQmlEngine 实例

    2. 为 QML 提供线程池。

    3. 可以处理 QJsonValueQAbstractListModel。并且是线程安全的。



  • @MidoriYakumo 嘿嘿~英语不好,不过大体功能如下:

    1. 一个线程一个 QQmlEngine 实例

    2. 为 QML 提供线程池。

    3. 可以处理 QJsonValue 和 QAbstractListModel。并且是线程安全的。



  • 代码进行了更新 thread_in_qml,变得更简洁了。不过还没有实现对 ListModel 的支持。


 

最近的回复

  • ![0_1540211558178_20181022_202905 (1).gif](正在上传 100%)

    阅读更多
  • 随后在git for windows这个软件中输入

    git clone git@git.coding.net:jiangcaiyang1/qtdreamrelease_windows.git

    就可以获得一个空的项目了。其它的项目也是同样的操作方法。然后呢,将此作为自动构建的代码库吧。
    此次介绍就是这样~😘 😘 🐧

    阅读更多
  • 0_1540027445466_很好.jpg
    萌梦安装包的分发始终是一个难题。要有一个稳定并且有效的方法进行分包,才能让大家都能够享受到萌梦动作编辑器的最新功能。我们目前有两种分发的方法:

    离线安装包。通常的位置是
    1.1 百度云盘
    1.2 QQ群文件
    1.3 github
    其中百度云盘和QQ群文件不会定期地更新,只有github的文件会定期更新。因为构建脚本要求每次有更改,都让github的文件更新。

    在线安装包。通常的位置是
    2.1 百度云盘
    2.2 QQ群文件
    2.3 github
    这里在线安装包的代码库主要还是在github中,在线安装包非常依赖CDN加速。目前github这块儿在国内做得不太好,还是希望在国内找一个类似的加速。目前可以选的只有两家:

    码云

    coding

    因为以前做过coding的加速,目前coding已经和腾讯云战略合作,所有的代码都要迁移到腾讯云了。而且对于代码库的空间要求变高了。128M。目前我的一个账号显然满足不了要求。所以有必要再注册一个账号。

    下面记录一下新建coding代码库的方法,并且尝试将萌梦安装包的线上代码库部分移动到coding代码库中。

    注册

    到这个位置注册。

    https://coding.net/register

    相信很快就会带你跳转到腾讯云中。不过这里我选择的是邮箱的方式,因为我希望注册尽量多的账号。

    设置ssh密钥

    进入控制台后,是这样的:
    0_1540028446495_2eeb7e93-c15b-4c82-9b36-58e1c304ea3f-image.png
    这个时候,我们要先创建项目,然后ssh的公钥。
    0_1540028822318_b3e81b93-099c-42d5-820d-deaf8d445913-image.png

    创建以后,开始部署公钥。在这里:
    0_1540028925337_d8059b64-8535-45e5-ba4c-d7454505aa5a-image.png
    这期间最好用OpenSSL生成SSH密钥。
    我使用的是git for windows来生成密钥的。方法是

    ssh-keygen

    如果你生成好了密钥,可以直接打开密钥内容看看:

    cat ~/.ssh/id_rsa.pub

    然后粘贴到网站的文本框中。

    添加好了之后,勾选“准许推送权限”,需要推送代码到代码库中。

    阅读更多

关注我们

微博
QQ群











召唤伊斯特瓦尔