博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Qt多线程间信号槽传递非QObject类型对象的参数
阅读量:4108 次
发布时间:2019-05-25

本文共 2247 字,大约阅读时间需要 7 分钟。

转自
http://qimo601.iteye.com/blog/1673578

一、以前就发现过这个问题:

在Qt项目中,有时候为了让自己的类,可以重载操作符 '=','<<','>>'. 也有时候需要用一个类进行文件的读写,所以很多C++类还是要简单化的,不需要继承QObject,不需要Qt的元对象机制。

但是对于这些简单C++类,有些时候要是调用Qt的信号槽当做参数进行跨线程发送,就会出现如下问题:

这种情况一般,编译可以通过,但会出现如下提示。

 

 

Cpp代码  
  1. QOBject::connect:Cannot queue arguments of type 'MoSystemLog'  
  2. (Make sure 'MoSystemLog' is registed using qRegisterMetaType().)  

 

意思是说,信号槽队列中的数据类型必须是系统能识别的元类型,不然得用qRegisterMetaType()进行注册。

 

二、解决方法:

第一种注册法:qRegisterMetatType<MoSystemLog>("MoSystemLog")

第二种修改Connect,加一个属性Qt::directConnection.

connect(cm, SIGNAL(sendLog(QUuid, QByteArray, bool)),

 this,SLOT(sendRes(QUuid,QByteArray,bool)), Qt::DirectConnection);

 

 

三、方法解释:

 

1、第一种解决方法,即使用排队方式的信号-槽机制,Qt的元对象系统(meta-object system)必须知道信号传递的参数类型。这样元系统可以将信号参数COPY下来,放在队列中等待事件唤醒,供槽函数调用。Just a note here, if you would have to pass custom data types between threads in Qt. As we know, a signal-slot connection is then (by default) of type Qt::QueuedConnection. Because in such a situation Qt needs to store passed parameters for a while, it creates their temporary copies. If it doesn’t recognize the passed data type, throws out an error:

2、第二种方法,直接调用对方槽函数,不需要保存参数。但是官方认为这样做有风险。

 

 

四、背景知识:

1、首先要了解enum Qt::ConnectionType

 

Qt支持6种连接方式,其中3种最主要:

Qt::DirectConnection(直连方式)

当信号发出后,相应的槽函数将立即被调用。emit语句后的代码将在所有槽函数执行完毕后被执行。(信号与槽函数关系类似于函数调用,同步执行)

Qt::QueuedConnection(排队方式)

当信号发出后,排队到信号队列中,需等到接收对象所属线程的事件循环取得控制权时才取得该信号,调用相应的槽函数。emit语句后的代码将在发出信号后立即被执行,无需等待槽函数执行完毕。(此时信号被塞到信号队列里了,信号与槽函数关系类似于消息通信,异步执行)

Qt::AutoConnection(自动方式)

Qt的默认连接方式,如果信号的发出和接收这个信号的对象同属一个线程,那个工作方式与直连方式相同;否则工作方式与排队方式相同。

Qt官方文档中写明:

With queued connections, the parameters must be of types that are known to Qt's meta-object system, because Qt needs to copy the arguments to store them in an event behind the scenes. If you try to use a queued connection and get the error message

QObject::connect: Cannot queue arguments of type 'MyType'

call() to register the data type before you establish the connection.

注意上面紫色的内容。


connect的第五个参数用于指定反应速度:
若将:
connect(cm, SIGNAL(sendLog(QUuid, QByteArray, bool)),
            this,SLOT(sendRes(QUuid,QByteArray,bool)));
改为:
connect(cm, SIGNAL(sendLog(QUuid, QByteArray, bool)),
            this,SLOT(sendRes(QUuid,QByteArray,bool)), Qt::DirectConnection);

可解决因信号没有及时发送,致使connect的接收方的槽不能作进一步处理,但是有风险

 

 

参考文章:

1、

2、

3、

转载地址:http://uzvsi.baihongyu.com/

你可能感兴趣的文章
web.py 0.3 新手指南 - 实时语言切换
查看>>
web.py 0.3 新手指南 - RESTful doctesting using app.request
查看>>
web.py 0.3 新手指南 - 使用db.query进行高级数据库查询
查看>>
web.py 0.3 新手指南 - 多数据库使用
查看>>
一步步开发 Spring MVC 应用
查看>>
「译」C++ 中 0 是十进制还是八进制?
查看>>
python: extend (扩展) 与 append (追加) 的差别
查看>>
「译」在 python 中,如果 x 是 list,为什么 x += "ha" 可以运行,而 x = x + "ha" 却抛出异常呢?...
查看>>
谷歌阅读器将于2013年7月1日停止服务,博客订阅转移到邮箱
查看>>
怎么成为技术大牛
查看>>
浅谈JavaScript的语言特性
查看>>
老生常谈:面试算法有必要吗?
查看>>
老生常谈:注释怎么写?
查看>>
「译」javascript 中的 delete
查看>>
「译」javascript 的 12 个怪癖(更新到第二篇)
查看>>
mixin 模式看起来好像装饰器模式,究竟是一个什么样的模式
查看>>
「译」JavaScript 的怪癖 1:隐式类型转换
查看>>
为什么说任何基于比较的算法将 5 个元素排序都需要 7 次?
查看>>
JavaScript 中的“自动分号插入”机制(ASI)
查看>>
javascript 中强制执行 toString()
查看>>