一、目录

转载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. DownLoadManager[20530:228829] DiskImageCache: Could not resolve the absolute path of the old directory.

    uiwebview 模拟器打开PDF文件时崩溃.报下面错误,还不知道为什么 DownLoadManager[20530:228829] DiskImageCache: Could not resolv ...

  2. noip刷题记录 20170823

    独木桥 怎么说呢 #include<iostream> #include<cstdio> #include<algorithm> using namespace s ...

  3. 【BZOJ 1024】 [SCOI2009]生日快乐

    [题目链接]:http://www.lydsy.com/JudgeOnline/problem.php?id=1024 [题意] [题解] 要求恰好分成n个部分;每个部分的面积都一样; 则dfs的时候 ...

  4. ue 不能初始化ftp组件

    进入到ue的安装目录下shift并右键-->在此处打开命令窗口,执行以下命令 regsvr32.exe wodFtpDLX.dll,如果是64位要执行regsvr32.exe wodFtpDLX ...

  5. 【16.52%】【codeforces 733C】Epidemic in Monstropolis

    time limit per test1 second memory limit per test256 megabytes inputstandard input outputstandard ou ...

  6. Spring实战5-基于Spring构建Web应用

    主要内容 将web请求映射到Spring控制器 绑定form参数 验证表单提交的参数 写在前面:关于Java Web,首先推荐一篇文章——写给java web一年左右工作经验的人,这篇文章的作者用精练 ...

  7. 经典书单 —— 语言/算法/机器学习/深度学习/AI/CV/PGM

    0.0 计算机科学 <Lex 与 Yacc> Think Complexity(使用 Python 语言) GitHub - AllenDowney/ThinkComplexity: Co ...

  8. Android 获取imei号码,获取手机型号和系统版本号

    在AndroidManifest.xml文件中要添加 <uses-permission android:name="android.permission.READ_PHONE_STAT ...

  9. [Servlet]Servlet工作流程及注意事项

    Servlet工作过程 采用Servlet完成Web实际的工作流应用程序是通过Tomcatserver公布服务,client与server遵循的端部之间的相互作用Http议完毕的. 详细工作流程例如以 ...

  10. virtualbox下ubuntu共享文件夹自动挂载

    1.若想删除挂载,可执行命令 umount -f /mnt/share 2.若想开机自动挂载,可以在 /etc/fstab 文件末添加一项     sharing /mnt/share vboxsf  ...