Qt线程间共享数据主要有两种方式:

  • 使用共享内存。即使用一个两个线程都能够共享的变量(如全局变量),这样两个线程都能够访问和修改该变量,从而达到共享数据的目的;
  • 使用singal/slot机制,把数据从一个线程传递到另外一个线程。

第一种办法在各个编程语言都使用普遍,而第二种方式倒是QT的特有方式,下面主要学习一下这种方式:

在线程之间传递signal与在一个线程内传递signal是不一样的。在一个线程内传递signal时,emit语句会直接调用所有连接的slot并等待到所有slot被处理完;在线程之间传递signal时,slot会被放到队列中(queue),而emit这个signal后会马上返回;默认情况,线程之间使用queue机制,而线程内使用direct机制,但在connect中可以改变这些默认的机制。

  1. //TextDevice.h
  2. #ifndef TEXTDEVICE_H
  3. #define TEXTDEVICE_H
  4. #include <QThread>
  5. #include <QString>
  6. #include <QMutex>
  7. class TextDevice : public QThread {
  8. Q_OBJECT
  9. public:
  10. TextDevice();
  11. void run();
  12. void stop();
  13. public slots:
  14. void write(const QString& text);
  15. private:
  16. int m_count;
  17. QMutex m_mutex;
  18. };
  19. #endif // TEXTDEVICE_H
  20. //TextDevice.cpp
  21. #include <QMutexLocker>
  22. #include <QDebug>
  23. #include <QString>
  24. #include "TextDevice.h"
  25. TextDevice::TextDevice() {
  26. m_count = 0;
  27. }
  28. void TextDevice::run() {
  29. exec();
  30. }
  31. void TextDevice::stop() {
  32. quit();
  33. }
  34. void TextDevice::write(const QString& text) {
  35. QMutexLocker locker(&m_mutex);
  36. qDebug() << QString("Call %1: %2").arg(m_count++).arg(text);
  37. }
  38. //TextThread.h
  39. #ifndef TEXTTHREAD_H
  40. #define TEXTTHREAD_H
  41. #include <QThread>
  42. #include <QString>
  43. class TextThread : public QThread {
  44. Q_OBJECT
  45. public:
  46. TextThread(const QString& text);
  47. void run();
  48. void stop();
  49. signals:
  50. void writeText(const QString&);
  51. private:
  52. QString m_text;
  53. bool m_stop;
  54. };
  55. #endif // TEXTTHREAD_H
  56. //TextThread.cpp
  57. #include "TextThread.h"
  58. TextThread::TextThread(const QString& text) : QThread() {
  59. m_text = text;
  60. m_stop = false;
  61. }
  62. void TextThread::stop() {
  63. m_stop = true;
  64. }
  65. void TextThread::run() {
  66. while(!m_stop) {
  67. emit writeText(m_text);
  68. sleep(1);
  69. }
  70. }
  71. //main.cpp
  72. #include <QApplication>
  73. #include <QMessageBox>
  74. #include "TextDevice.h"
  75. #include "TextThread.h"
  76. int main(int argc, char** argv) {
  77. QApplication app(argc, argv);
  78. //启动线程
  79. TextDevice device;
  80. TextThread foo("foo"), bar("bar");
  81. //把两个线程使用signal/slot连接起来
  82. QObject::connect(&foo, SIGNAL(writeText(const QString&)), &device, SLOT(write(const QString&)));
  83. QObject::connect(&bar, SIGNAL(writeText(const QString&)), &device, SLOT(write(const QString&)));
  84. //启动线程
  85. foo.start();
  86. bar.start();
  87. device.start();
  88. QMessageBox::information(0, "Threading", "Close me to stop.");
  89. //停止线程
  90. foo.stop();
  91. bar.stop();
  92. device.stop();
  93. //等待线程结束
  94. device.wait();
  95. foo.wait();
  96. bar.wait();
  97. return 0;
  98. }

上面例子代码可以看出两个线程之间传送了类型为QString的信息。像QString等这些QT本身定义的类型,直接传送即可。但如果是自己定义的类型如果想使用signal/slot来传递的话,则没有这么简单。直接使用的话,会产生下面这种错误:
          QObject::connect: Cannot queue arguments of type 'TextAndNumber' (Make sure 'TextAndNumber' is registed using qRegisterMetaType().)
         原因:当一个signal被放到队列中(queued)时,它的参数(arguments)也会被一起一起放到队列中(queued起来),这就意味着参数在被传送到slot之前需要被拷贝、存储在队列中(queue)中;为了能够在队列中存储这些参数(argument),Qt需要去construct、destruct、copy这些对象,而为了让Qt知道怎样去作这些事情,参数的类型需要使用qRegisterMetaType来注册(如错误提示中的说明)

步骤:(以自定义TextAndNumber类型为例)

  • 自定一种类型,在这个类型的顶部包含:#include <QMetaType>
  • 在类型定义完成后,加入声明:Q_DECLARE_METATYPE(TextAndNumber);
  • 在main()函数中注册这种类型:qRegisterMetaType<TextAndNumber>("TextAndNumber");
  • 如果还希望使用这种类型的引用,可同样要注册:qRegisterMetaType<TextAndNumber>("TextAndNumber&");
  1. //TextAndNumber.h
  2. #ifndef TEXTANDNUMBER_H
  3. #define TEXTANDNUMBER_H
  4. #include <QMetaType>
  5. //必须包含QMetaType,否则会出现下面错误:
  6. //error: expected constructor, destructor, or type conversion before ‘;’ token
  7. #include <QString>
  8. class TextAndNumber {
  9. public:
  10. TextAndNumber();
  11. TextAndNumber(int, QString);
  12. int count();
  13. QString text();
  14. private:
  15. int m_count;
  16. QString m_text;
  17. };
  18. Q_DECLARE_METATYPE(TextAndNumber);
  19. #endif // TEXTANDNUMBER_H
  20. //TextAndNumber.cpp
  21. #include "TextAndNumber.h"
  22. TextAndNumber::TextAndNumber() {
  23. }
  24. TextAndNumber::TextAndNumber(int count, QString text) {
  25. m_count = count;
  26. m_text = text;
  27. }
  28. int TextAndNumber::count() {
  29. return m_count;
  30. }
  31. QString TextAndNumber::text() {
  32. return m_text;
  33. }
  34. //TextDevice.h
  35. #ifndef TEXTDEVICE_H
  36. #define TEXTDEVICE_H
  37. #include <QThread>
  38. #include <QDebug>
  39. #include <QString>
  40. #include "TextAndNumber.h"
  41. class TextDevice : public QThread {
  42. Q_OBJECT
  43. public:
  44. TextDevice();
  45. void run();
  46. void stop();
  47. public slots:
  48. void write(TextAndNumber& tran);
  49. private:
  50. int m_count;
  51. };
  52. #endif // TEXTDEVICE_H
  53. //TextDevice.cpp
  54. #include "TextDevice.h"
  55. TextDevice::TextDevice() : QThread() {
  56. m_count = 0;
  57. }
  58. void TextDevice::run() {
  59. exec();
  60. }
  61. void TextDevice::stop() {
  62. quit();
  63. }
  64. void TextDevice::write(TextAndNumber& tran) {
  65. qDebug() << QString("Call %1 (%3): %2").arg(m_count++).arg(tran.text()).arg(tran.count());
  66. }
  67. //TextThread.h
  68. #ifndef TEXTTHREAD_H
  69. #define TEXTTHREAD_H
  70. #include <QThread>
  71. #include <QString>
  72. #include "TextAndNumber.h"
  73. class TextThread : public QThread {
  74. Q_OBJECT
  75. public:
  76. TextThread(const QString& text);
  77. void run();
  78. void stop();
  79. signals:
  80. void writeText(TextAndNumber& tran);
  81. private:
  82. QString m_text;
  83. int m_count;
  84. bool m_stop;
  85. };
  86. #endif // TEXTTHREAD_H
  87. //TextThread.cpp
  88. #include "TextThread.h"
  89. TextThread::TextThread(const QString& text) : QThread() {
  90. m_text = text;
  91. m_stop = false;
  92. m_count = 0;
  93. }
  94. void TextThread::run() {
  95. while(!m_stop) {
  96. TextAndNumber tn(m_count++, m_text);
  97. emit writeText(tn);
  98. sleep(1);
  99. }
  100. }
  101. void TextThread::stop() {
  102. m_stop = true;
  103. }
  104. //main.cpp
  105. #include <QApplication>
  106. #include <QMessageBox>
  107. #include "TextThread.h"
  108. #include "TextDevice.h"
  109. #include "TextAndNumber.h"
  110. int main(int argc, char *argv[])
  111. {
  112. QApplication app(argc, argv);
  113. qRegisterMetaType<TextAndNumber>("TextAndNumber");
  114. qRegisterMetaType<TextAndNumber>("TextAndNumber&");
  115. TextDevice device;
  116. TextThread foo("foo"), bar("bar");
  117. QObject::connect(&foo, SIGNAL(writeText(TextAndNumber&)), &device, SLOT(write(TextAndNumber&)));
  118. QObject::connect(&bar, SIGNAL(writeText(TextAndNumber&)), &device, SLOT(write(TextAndNumber&)));
  119. device.start();
  120. foo.start();
  121. bar.start();
  122. QMessageBox::information(0, "Threading", "Click me to close");
  123. foo.stop();
  124. bar.stop();
  125. device.stop();
  126. foo.wait();
  127. bar.wait();
  128. device.wait();
  129. qDebug() << "Application end.";
  130. return 0;
  131. }

http://blog.csdn.net/swingline/article/details/5635288

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

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

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

  2. Qt 学习之路:深入 Qt5 信号槽新语法

    在前面的章节(信号槽和自定义信号槽)中,我们详细介绍了有关 Qt 5 的信号槽新语法.由于这次改动很大,许多以前看起来不是问题的问题接踵而来,因此,我们用单独的一章重新介绍一些 Qt 5 的信号槽新语 ...

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

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

  4. 详解 Qt 线程间共享数据(使用signal/slot传递数据,线程间传递信号会立刻返回,但也可通过connect改变)

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

  5. Disruptor 线程间共享数据无需竞争

    队列的作用是缓冲 缓冲到 队列的空间里.. 线程间共享数据无需竞争 原文 地址  作者  Trisha   译者:李同杰 LMAX Disruptor 是一个开源的并发框架,并获得2011 Duke’ ...

  6. Qt 学习之路 2(14):对话框数据传递

    Home / Qt 学习之路 2 / Qt 学习之路 2(14):对话框数据传递 Qt 学习之路 2(14):对话框数据传递  豆子  2012年9月15日  Qt 学习之路 2  53条评论 对话框 ...

  7. ThreadLocal实现session中用户信息 的线程间共享(精)

    ThreadLocal并不难理解,我总结的最简单的理解就是: ThreadLocal像其它变量一样(局部.全局.静态)也是一种变量类型,只是他是线程变量,更直白的说他是一种变量作用域,即他的作用域是当 ...

  8. ThreadLocal实现session中用户信息 的线程间共享

    转载自:http://blog.sina.com.cn/s/blog_4b5bc01101013gok.html ThreadLocal并不难理解,我总结的最简单的理解就是: ThreadLocal像 ...

  9. Java并发基础09. 多个线程间共享数据问题

    先看一个多线程间共享数据的问题: 设计四个线程,其中两个线程每次对data增加1,另外两个线程每次对data减少1. 从问题来看,很明显涉及到了线程间通数据的共享,四个线程共享一个 data,共同操作 ...

随机推荐

  1. Qt socket中怎么传结构体?

    直接发送和接收结构体,例如:struct A {...};struct A objectA; 发送的时候: tcpSocket->write((char *)&objectA, size ...

  2. HDU 1090 A+B for Input-Output Practice (II)

    #include <cstdio> int main() { int n,a,b; scanf("%d",&n); ; i<=n; i++) { scan ...

  3. PLSQL 导入表到Oracle------》从一个表空间导入到其它表空间

        在用PLSQL导入.dmp文件到Oracle时出现的问题如下: Import started on 2015/11/18 10:42:44E:\oracle\product\10.2.0\db ...

  4. Java编程思想重点笔记

    首先声明转自https://github.com/lanxuezaipiao/ReadingNotes 无意中发现,写的蛮好转过来学习下. 1. Java中的多态性理解(注意与C++区分) Java中 ...

  5. JavaScript 覆盖document.createElement 方法 解决window.close在火狐下不兼容问题)

    近期项目遇到了问题,有个asp.net web程序仅仅能在IE7 执行.如今xp都淘汰了,大家都用IE8-IE11,因此这个web app也须要升级 适应全部IE版本号.照成IE版本号不兼容的问题主要 ...

  6. HDU 1796 容斥原理 How many integers can you find

    题目连接   http://acm.hdu.edu.cn/showproblem.php?pid=1796 处男容斥原理  纪念一下  TMD看了好久才明白DFS... 先贴代码后解释 #includ ...

  7. 关于var(string)++的类型自动转换

    展示时间: var miao="50"; var fen="59"; var shi="00"; setInterval(fun, 1000 ...

  8. 简单JSONP跨域请求

    JSONP原理:利用<script>标签的src属性实现跨域的请求.可在URL中提供回调函数的名字.后台进过处理后将数据以回调函数参数的形式返回. demo:JSONP请求不同端口的数据 ...

  9. 在TextBox里面仅仅允许数字,按Enter键进入下一个TextBox

    <html xmlns="http://www.w3.org/1999/xhtml"> <head> <title></title> ...

  10. Unity 对象池 生产 保存

    Unity对象池主要是保存那些常用的物体,避免他们在不断销毁和创造中损坏性能. 主要思路为:创造物体时,判断是否存在,如果存在则调用并使其显示.如果不存在则创造一个新的. 当销毁时,调用协程延时隐藏物 ...