一、目录

转载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++标准库中大名鼎鼎的函数对象。

  1. class FunctionTransfer : public QObject
  2. {
  3. Q_OBJECT
  4. public:
  5. ///@brief 构造函数
  6. explicit FunctionTransfer(QObject *parent = 0);
  7. public:
  8. ///@brief 制定函数f在main中执行
  9. static void execinmain(std::tr1::function<void()> f);
  10. signals:
  11. ///@brief 在别的线程有函数对象传来
  12. void comming(std::tr1::function<void()> f);
  13. public slots:
  14. ///@brief 执行函数对象
  15. void exec(std::tr1::function<void()> f);
  16. };

然后是源文件:

  1. //在全局数据区实例化一个FunctionTransfer的实例,该实例所在的县城就是主线程。
  2. FunctionTransfer main_thread_forward;
  3. void FunctionTransfer::execinmain(std::tr1::function<void()> f)
  4. {
  5. main_thread_forward.exec(f);
  6. }
  7. FunctionTransfer::FunctionTransfer(QObject *parent) :
  8. QObject(parent)
  9. {
  10. connect(this,SIGNAL(comming(std::tr1::function<void()>)),this,SLOT(exec(std::tr1::function<void()>)),Qt::BlockingQueuedConnection);
  11. }
  12. void FunctionTransfer::exec(std::tr1::function<void()> f)
  13. {
  14. if(Gt::isMainThread())
  15. {
  16. f();
  17. }
  18. else
  19. {
  20. emit this->comming(f);
  21. }
  22. }

非常简单的逻辑,如果在主线程就执行,如果不是在主线程就发给主线程,主线程接到之后就执行。
类有了,接下来考虑实用的场合,比如有一个类 A,A有个方法f不能再后台执行,需要跑到前台,怎么办呢,上代码:

  1. 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 点击打开链接

  1. #pragma once
  2. #include <qthread.h>
  3. #include <functional>
  4. class QIoService : QObject
  5. {
  6. public:
  7. QIoService(bool startinthread)
  8. {
  9. if(startinthread)
  10. {
  11. worker=new QThread(NULL);
  12. worker->start();
  13. this->moveToThread(worker);
  14. }
  15. else
  16. {
  17. //this object is created in create thread!!!
  18. }
  19. }
  20. void post(std::function<void()> func);
  21. void send(std::function<void()> func);
  22. void post(std::function<void()> func,int ms);
  23. void send(std::function<void()> func,int ms);
  24. virtual bool event ( QEvent * e);
  25. protected:
  26. QThread *worker;
  27. };
  28. //this should run in mainthread
  29. extern QIoService *main_ioservice;
  30. #include "stdafx.h"
  31. #include "qioservice.h"
  32. #include <qapplication.h>
  33. #include <qtconcurrentrun.h>
  34. QIoService *main_ioservice=NULL;
  35. class FunctionEvent : public QEvent
  36. {
  37. public:
  38. static const QEvent::Type myType = static_cast<QEvent::Type>(2000);
  39. FunctionEvent(std::function<void()> f)
  40. :QEvent(myType)
  41. {
  42. func=f;
  43. }
  44. ~FunctionEvent()
  45. {
  46. //这个他会自动删除
  47. }
  48. std::function<void()> func;
  49. };
  50. void QIoService::post(std::function<void()> func)
  51. {
  52. QApplication::instance()->postEvent(this,new FunctionEvent(func));
  53. }
  54. void QIoService::send(std::function<void()> func)
  55. {
  56. QApplication::instance()->sendEvent(this,new FunctionEvent(func));
  57. }
  58. void QIoService::post(std::function<void()> func,int ms)
  59. {
  60. auto lam = [&]()
  61. {
  62. QThread::currentThread()->wait(ms);
  63. post(func);
  64. };
  65. QtConcurrent::run(lam);
  66. }
  67. void QIoService::send(std::function<void()> func,int ms)
  68. {
  69. auto lam = [&]()
  70. {
  71. QThread::currentThread()->wait(ms);
  72. send(func);
  73. };
  74. QtConcurrent::run(lam);
  75. }
  76. bool QIoService::event ( QEvent * e)
  77. {
  78. if(e->type()==FunctionEvent::myType)
  79. {
  80. FunctionEvent *fe=(FunctionEvent *)e;
  81. fe->func();
  82. return true;
  83. //这个他会自动删除,不用我们自己手工delete
  84. }
  85. return false;
  86. }

注解:

  1. //比如你在另外一个线程,你收到数据,想修改界面。就弄个全局变量
  2. QIoService g_ui_ios(false);
  1. //你只要
  2. g_ui_ios.send([你的变量]
  3. {
  4. //修改界面数据,这个会在主线程执行
  5. });
  6. //如果你是想写个任务队列,
  7. QIoService g_worker_ios(true);
  1. //你要把某段事情丢到其他线程执行,就
  2. g_worker_ios.send([]
  3. {
  4. //这段会在其他线程执行。
  5. 如果执行完了,又想在主线程执行某段,这里可以继续
  6. //g_ui_ios.send([]
  7. {
  8. //这段会在主线程执行
  9. });
  10. });

三、更新

2016年8月27日 第一次更新

http://blog.csdn.net/qq2399431200/article/details/52335517

Qt 跨UI线程的数据交换和信号-槽调用实现方案汇总的更多相关文章

  1. [转]QT子线程与主线程的信号槽通信-亲测可用!

    近用QT做一个服务器,众所周知,QT的主线程必须保持畅通,才能刷新UI.所以,网络通信端采用新开线程的方式.在涉及到使用子线程更新Ui上的控件时遇到了点儿麻烦.网上提供了很多同一线程不同类间采用信号槽 ...

  2. Disruptor——一种可替代有界队列完成并发线程间数据交换的高性能解决方案

    本文翻译自LMAX关于Disruptor的论文,同时加上一些自己的理解和标注.Disruptor是一个高效的线程间交换数据的基础组件,它使用栅栏(barrier)+序号(Sequencing)机制协调 ...

  3. Exchanger实现线程间数据交换

    package com.duchong.concurrent; import java.util.ArrayList; import java.util.List; import java.util. ...

  4. 详解 Qt 线程间共享数据(用信号槽方式)

    使用共享内存.即使用一个两个线程都能够共享的变量(如全局变量),这样两个线程都能够访问和修改该变量,从而达到共享数据的目的. Qt 线程间共享数据是本文介绍的内容,多的不说,先来啃内容.Qt线程间共享 ...

  5. Qt学习:线程间共享数据(使用信号槽传递数据,必须提前使用qRegisterMetaType来注册参数的类型)

    Qt线程间共享数据主要有两种方式: 使用共享内存.即使用一个两个线程都能够共享的变量(如全局变量),这样两个线程都能够访问和修改该变量,从而达到共享数据的目的: 使用singal/slot机制,把数据 ...

  6. moveToThread的最简单用法(依葫芦画瓢即可)(使得线程也更偏向于信号槽的使用方法)

    /*! * \file main.cpp * * Copyright (C) 2010, dbzhang800 * All rights reserved. * */ #include <QtC ...

  7. WPF / Win Form:多线程去修改或访问UI线程数据的方法( winform 跨线程访问UI控件 )

    WPF:谈谈各种多线程去修改或访问UI线程数据的方法http://www.cnblogs.com/mgen/archive/2012/03/10/2389509.html 子线程非法访问UI线程的数据 ...

  8. Qt5中运行后台网络读取线程与主UI线程互交

    项目中有一个需求就是,因为需要请求服务端数据,因为网络的读取会阻塞,所以该过程不能放在Qt中的UI主线程当中,需要用一个后台线程来读取数据,数据准备完毕后 在通过Qt5中的信号槽机制来跨线程的传递数据 ...

  9. c#多线程(UI线程,控件显示更新) Invoke和BeginInvoke 区别

    如果只是直接使用子线程访问UI控件,直接看内容三,如果想深入了解从内容一看起. 一.Control.Invoke和BeginInvoke方法的区别 先上总结: Control.Invoke 方法 (D ...

随机推荐

  1. easyui样式及js导入顺序及刷新回车的问题

    在使用easyui时,需要导入样式表及其js文件,在导入时.不光要遵守jquery包在easyui包的前面,还需要把样式表放在js的前边 <link type="text/css&qu ...

  2. 配置SVN服务器

    svn启动: 版本控制对于团队合作显得尤为重要,那么如何在iOS开发中进行版本控制呢?在今天的博客中将会介绍如何在MAC下配置SVN服务器,如何导入我们的工程,如何在Xcode中进行工程的checkO ...

  3. Configure Two DataSources ---

    67. Data Access 67.1 Configure a DataSource To override the default settings just define a @Bean of ...

  4. Scala & IntelliJ IDEA环境搭建升级版:在JAVA中调用Scala的helloworld

    --------------------- 前言 --------------------- 项目关系,希望用Spark GraphX做数据分析及图像展示,但前提是得会spark:spark是基于sc ...

  5. 在echarts中自定义直方图bar上悬浮透明窗文本内容

    直接贴代码: <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <ti ...

  6. VS2005 编译 Qt4.8.2库,并修正源码中的错误

    从昨天开始利用VS2005对Qt4.8.2库进行编译,在编译到某个文件时,总是报错,提示VS的cl.exe和nmake.exe返回致命错误.错误罗列如下: DefaultLocalizationStr ...

  7. WPF4文字模糊不清晰、边框线条粗细不一致的解决方法

    原文:WPF4文字模糊不清晰.边框线条粗细不一致的解决方法 软件测试过程中发现在一台1600*900的分辨率电脑上文字模糊,甚至某些个文字出现压缩扭曲 经过实践,发现按下面方法能解决一点问题: 在窗口 ...

  8. 组态DNS、搜索域名和主机名

    一个.组态DNS和搜索领域 特别配置DNS该文件是/etc/resolv.conf:同时,我们可以ifcfg-eth0网卡的配置和其他配置文件中指定的. 演示样本: [root@instructor ...

  9. 【狼窝乀野狼】Windows Server 2008 R 配置 Microsoft Server 2008 远程登录连接

    如果你已经了解了,或者你已经经历了,那么此篇文章对你是毫无用处.因为文笔深处未必有自己亲身体验来的真实有效. 闲话少说,直接上菜. 最近脑子“抽筋”,想安装一个服务器来玩玩,那么怎么选择呢?我的PC是 ...

  10. matlab 工具函数 —— axnote(在坐标轴上写文本内容)

    function axnote(string) font_size = get(0, 'DefaultAxesFontSize'); if 1 h1 = text(0.99, 0.05, string ...