Qt 跨UI线程的数据交换和信号-槽调用实现方案汇总
一、目录
转载1: http://my.oschina.NET/fanhuazi/blog/737224?ref=myread 点击打开链接
转载2: http://www.qtcn.org/bbs/read-htm-tid-60505-ds-1-page-1.html#172183 点击打开链接
二、内容
由于以下两篇转载文章都使用了C++11 新特性“lamda表达式”,为了方便同仁阅读以下内容,在此引用一片文章对“C++11 lamda表达式”做一个简要介绍:
“C++11 lamda表达式” ,点击此链接先了解C++11新增语法,对后面内容的阅读会有帮助。
转载1: http://my.oschina.Net/fanhuazi/blog/737224?ref=myread 点击打开链接
在Qt中将函数发送到主线程执行
……(省略部分内容,完成内容请参看上面的原文链接)数据共享的问题,试想“后台线程(非UI线程)中的数据如何能够被前台(UI线程)所使用,而且前台后台不一定在一个类里面?把数据打包通过信号传给前台?”想想就是很麻烦的事情,难道每个这样的需求场合都要做一遍这样的事情吗?感谢时间,因为时间穿过2011年,C++的新标准已经完美的解决了这个问题,那就是函数对象。
Qt的4.8.6版本所使用的mingw4.9.2版本是支持C++11的,如果你用的是老掉牙的rhel5系统,则需要升级编译器了,因为C++11要在GCC 4.5以上的版本中才会支持。
首先我们定义一个类:FunctionTransfer(函数大挪移),这个类继承自QObject,并使用Q_OBJECT标签来使用信号槽机制。代码中的“std::tr1::function<void()>”就是C++标准库中大名鼎鼎的函数对象。
- class FunctionTransfer : public QObject
- {
- Q_OBJECT
- public:
- ///@brief 构造函数
- explicit FunctionTransfer(QObject *parent = 0);
- public:
- ///@brief 制定函数f在main中执行
- static void execinmain(std::tr1::function<void()> f);
- signals:
- ///@brief 在别的线程有函数对象传来
- void comming(std::tr1::function<void()> f);
- public slots:
- ///@brief 执行函数对象
- void exec(std::tr1::function<void()> f);
- };
然后是源文件:
- //在全局数据区实例化一个FunctionTransfer的实例,该实例所在的县城就是主线程。
- FunctionTransfer main_thread_forward;
- void FunctionTransfer::execinmain(std::tr1::function<void()> f)
- {
- main_thread_forward.exec(f);
- }
- FunctionTransfer::FunctionTransfer(QObject *parent) :
- QObject(parent)
- {
- connect(this,SIGNAL(comming(std::tr1::function<void()>)),this,SLOT(exec(std::tr1::function<void()>)),Qt::BlockingQueuedConnection);
- }
- void FunctionTransfer::exec(std::tr1::function<void()> f)
- {
- if(Gt::isMainThread())
- {
- f();
- }
- else
- {
- emit this->comming(f);
- }
- }
非常简单的逻辑,如果在主线程就执行,如果不是在主线程就发给主线程,主线程接到之后就执行。
类有了,接下来考虑实用的场合,比如有一个类 A,A有个方法f不能再后台执行,需要跑到前台,怎么办呢,上代码:
- FunctionTransfer::execinmain([this](){this->f();});
作为参数的lamda表达式捕获了类A的this指针,然后转换为C++的函数对象,然后跑到前台去执行了,执行完成后才会返回,是不是灰常简洁。
转载2:http://www.qtcn.org/bbs/read-htm-tid-60505-ds-1-page-1.html#172183 点击打开链接
- #pragma once
- #include <qthread.h>
- #include <functional>
- class QIoService : QObject
- {
- public:
- QIoService(bool startinthread)
- {
- if(startinthread)
- {
- worker=new QThread(NULL);
- worker->start();
- this->moveToThread(worker);
- }
- else
- {
- //this object is created in create thread!!!
- }
- }
- void post(std::function<void()> func);
- void send(std::function<void()> func);
- void post(std::function<void()> func,int ms);
- void send(std::function<void()> func,int ms);
- virtual bool event ( QEvent * e);
- protected:
- QThread *worker;
- };
- //this should run in mainthread
- extern QIoService *main_ioservice;
- #include "stdafx.h"
- #include "qioservice.h"
- #include <qapplication.h>
- #include <qtconcurrentrun.h>
- QIoService *main_ioservice=NULL;
- class FunctionEvent : public QEvent
- {
- public:
- static const QEvent::Type myType = static_cast<QEvent::Type>(2000);
- FunctionEvent(std::function<void()> f)
- :QEvent(myType)
- {
- func=f;
- }
- ~FunctionEvent()
- {
- //这个他会自动删除
- }
- std::function<void()> func;
- };
- void QIoService::post(std::function<void()> func)
- {
- QApplication::instance()->postEvent(this,new FunctionEvent(func));
- }
- void QIoService::send(std::function<void()> func)
- {
- QApplication::instance()->sendEvent(this,new FunctionEvent(func));
- }
- void QIoService::post(std::function<void()> func,int ms)
- {
- auto lam = [&]()
- {
- QThread::currentThread()->wait(ms);
- post(func);
- };
- QtConcurrent::run(lam);
- }
- void QIoService::send(std::function<void()> func,int ms)
- {
- auto lam = [&]()
- {
- QThread::currentThread()->wait(ms);
- send(func);
- };
- QtConcurrent::run(lam);
- }
- bool QIoService::event ( QEvent * e)
- {
- if(e->type()==FunctionEvent::myType)
- {
- FunctionEvent *fe=(FunctionEvent *)e;
- fe->func();
- return true;
- //这个他会自动删除,不用我们自己手工delete
- }
- return false;
- }
注解:
- //比如你在另外一个线程,你收到数据,想修改界面。就弄个全局变量
- QIoService g_ui_ios(false);
- //你只要
- g_ui_ios.send([你的变量]
- {
- //修改界面数据,这个会在主线程执行
- });
- //如果你是想写个任务队列,
- QIoService g_worker_ios(true);

- //你要把某段事情丢到其他线程执行,就
- g_worker_ios.send([]
- {
- //这段会在其他线程执行。
- 如果执行完了,又想在主线程执行某段,这里可以继续
- //g_ui_ios.send([]
- {
- //这段会在主线程执行
- });
- });
三、更新
2016年8月27日 第一次更新
http://blog.csdn.net/qq2399431200/article/details/52335517
Qt 跨UI线程的数据交换和信号-槽调用实现方案汇总的更多相关文章
- [转]QT子线程与主线程的信号槽通信-亲测可用!
近用QT做一个服务器,众所周知,QT的主线程必须保持畅通,才能刷新UI.所以,网络通信端采用新开线程的方式.在涉及到使用子线程更新Ui上的控件时遇到了点儿麻烦.网上提供了很多同一线程不同类间采用信号槽 ...
- Disruptor——一种可替代有界队列完成并发线程间数据交换的高性能解决方案
本文翻译自LMAX关于Disruptor的论文,同时加上一些自己的理解和标注.Disruptor是一个高效的线程间交换数据的基础组件,它使用栅栏(barrier)+序号(Sequencing)机制协调 ...
- Exchanger实现线程间数据交换
package com.duchong.concurrent; import java.util.ArrayList; import java.util.List; import java.util. ...
- 详解 Qt 线程间共享数据(用信号槽方式)
使用共享内存.即使用一个两个线程都能够共享的变量(如全局变量),这样两个线程都能够访问和修改该变量,从而达到共享数据的目的. Qt 线程间共享数据是本文介绍的内容,多的不说,先来啃内容.Qt线程间共享 ...
- Qt学习:线程间共享数据(使用信号槽传递数据,必须提前使用qRegisterMetaType来注册参数的类型)
Qt线程间共享数据主要有两种方式: 使用共享内存.即使用一个两个线程都能够共享的变量(如全局变量),这样两个线程都能够访问和修改该变量,从而达到共享数据的目的: 使用singal/slot机制,把数据 ...
- moveToThread的最简单用法(依葫芦画瓢即可)(使得线程也更偏向于信号槽的使用方法)
/*! * \file main.cpp * * Copyright (C) 2010, dbzhang800 * All rights reserved. * */ #include <QtC ...
- WPF / Win Form:多线程去修改或访问UI线程数据的方法( winform 跨线程访问UI控件 )
WPF:谈谈各种多线程去修改或访问UI线程数据的方法http://www.cnblogs.com/mgen/archive/2012/03/10/2389509.html 子线程非法访问UI线程的数据 ...
- Qt5中运行后台网络读取线程与主UI线程互交
项目中有一个需求就是,因为需要请求服务端数据,因为网络的读取会阻塞,所以该过程不能放在Qt中的UI主线程当中,需要用一个后台线程来读取数据,数据准备完毕后 在通过Qt5中的信号槽机制来跨线程的传递数据 ...
- c#多线程(UI线程,控件显示更新) Invoke和BeginInvoke 区别
如果只是直接使用子线程访问UI控件,直接看内容三,如果想深入了解从内容一看起. 一.Control.Invoke和BeginInvoke方法的区别 先上总结: Control.Invoke 方法 (D ...
随机推荐
- 学习鸟哥的Linux私房菜笔记(4)——文件
一.检查文件 用ls -l以长模式查看文件的详细信息,包含当前目录的硬盘使用空间.文件类型.文件权限.硬连接数.文件拥有者.文件所属组.文件大小.更改时间.文件名称. 用file检查文件类型 由于li ...
- 与Boss大雷探讨JavaWeb开发、电商与网络安全
最近几个月,与公司Boss大雷交流得比较多,也学习到了很多新的东西,了解到了一些没有接触和实践的业界做法. 简要介绍下Boss,姓雷,定居武汉好几年了,之前在一号店.UC.支付宝干过,有丰富的电商-支 ...
- winfrom 操作PPT
///winfrom 操作PPT using System; using System.Collections.Generic; using System.Linq; using System.Tex ...
- Spire.Doc组件
使用Spire.Doc组件利用模板导出Word文档 以前一直是用Office的组件实现Word文档导出,但是让客户在服务器安装Office,涉及到版权:而且Office安装,包括权限配置也是比较麻烦. ...
- freemarker中间split字符串切割
freemarker中间split字符串切割 1.简易说明 split切割:用来依据另外一个字符串的出现将原字符串切割成字符串序列 2.举例说明 <#--freemarker中的split字符串 ...
- Web自动化测试(全网最给力自动化教程)
http://www.cnblogs.com/zidonghua/p/7430083.html python+selenium自动化软件测试(第2章):WebDriver API 欢迎您来阅读和练手! ...
- Java_压缩与解压工具类
转载请注明出处:http://blog.csdn.net/y22222ly/article/details/52201675 zip压缩,解压 zip压缩与解压主要依靠java api的两个类: Zi ...
- sql 连接查询
什么是连接查询呢 概念:根据两个表或多个表的列之间的关系,从这些表中查询数据. 目的:实现多个表查询操作. 分类 首先划分一下,连接分为三种:内连接.外连接.交叉连接 内连接(INNER JOIN): ...
- webpack优化经验1(持续)
1 不知道该优化哪里 先开启gzip压缩,这样可以很直接的减少请求包的体积,效果显著,不过需要在服务器端作相应的配置才能生效 2拆分vendor包, 减少单体包的体积,并行加载 通过配置,将不同的公用 ...
- CUDA中的常量内存__constant__
GPU包含数百个数学计算单元,具有强大的处理运算能力,可以强大到计算速率高于输入数据的速率,即充分利用带宽,满负荷向GPU传输数据还不够它计算的.CUDA C除全局内存和共享内存外,还支持常量内存,常 ...