QT信号槽实现分析

1.宏定义

qt中引入了MOC来反射,编译阶段变成 MOC-->预处理-->编译-->汇编-->链接

1-1、Q_OBJECT

这个宏定义了一系列代码,包括元对象和处理的函数

cpp 复制代码
#define Q_OBJECT \
	public: \
	    QT_WARNING_PUSH \
	    Q_OBJECT_NO_OVERRIDE_WARNING \
	    static const QMetaObject staticMetaObject; \
	    virtual const QMetaObject *metaObject() const; \
	    virtual void *qt_metacast(const char *); \
	    virtual int qt_metacall(QMetaObject::Call, int, void **); \
	    QT_TR_FUNCTIONS \
	private: \
	    Q_OBJECT_NO_ATTRIBUTES_WARNING \
	    Q_DECL_HIDDEN_STATIC_METACALL static void qt_static_metacall(QObject *, 		QMetaObject::Call, int, void **); \
	    QT_WARNING_POP \
	    struct QPrivateSignal {}; \
	    QT_ANNOTATE_CLASS(qt_qobject, "")

1-2、signal slot emit

从如下得知,slot和emit只是作一个修饰而已,本身为空没有任何作用,signal也只是声明为pulic

cpp 复制代码
#define signal public
#define slot
#define emit

1-2、qt4中诞生的SIGANL SLOT

可以看到这两个宏也只是把我们输入的函数转成了字符串,后续去匹配

cpp 复制代码
#define SIGANL "2"#a
#define SLOT "1"#a

我们去VS项目--属性--C/C+±-预处理器--预处理文件生成选择:是

发现预编译后宏展开如下。即验证上述

2.信号槽规则

1.信号的参数可以比槽多,反之不行,信号槽可以一对一,一对多,多对一

2.槽函数执行的顺序是按连接时的顺序依次执行的,重复connect会导致多次执行

cpp 复制代码
connect(obj, SIGNAL(sig1), this, SLOT(slot1()));
connect(obj, SIGNAL(sig1), this, SLOT(slot2()));
那么sig1发出时,先执行槽函数 slot1, 再执行槽函数 slot2

3.如果用SIGNAL方式connect,可以不需要作用域,因为上面说过了,他把他转为了字符串。目前遇到好处是可以绑定多态信号槽,而无需引入派生类文件。坏处是不会进行检查,编译时才报warning

qt5的方式connect就需要,好处是写代码时就会进行参数匹配检查

4.连接可以被disconnect删除

5.注意connect第五个参数,详细可以见QT第五个参数,举两个常见例子

在子线程抛出来信号,主线程若不绑定为QueuedConnection或BlockingQueuedConnection,会接收不到。

如果用BlockingQueuedConnection,信号槽不能在同一个线程因为发送完信号后发送者所在线程会阻塞,直到槽函数运行完。本来就阻塞了。更别谈槽函数运行了。死锁了

3.connect实现

connect实现代码非常多,感兴趣的可以到这里阅读:QT5 qobject.cpp

总结下来分为这几步:

1.QObject类对象内部维护了一个名为connectionLists的成员变量,用于记录信号和槽函数的关联

2.MOC 会在预处理阶段根据 Q_OBJECT 宏生成对应的元对象信息,并将这些信息存储在 QObject 类中的一个指针变量中,可以通过 QObject::metaObject() 方法获取

3.在发射信号时,信号方会生成一个携带信号索引和参数的结构体,然后调用 QMetaObject::activate() 方法。QMetaObject::activate() 方法会先根据槽函数所在的类的元对象信息,获取该类存储的槽函数索引,然后根据索引找到相应的槽函数并调用

相关推荐
唐墨1239 小时前
android与Qt类比
android·开发语言·qt
道剑剑非道10 小时前
QT开发技术【ffmpeg + QAudioOutput】音乐播放器 完善
开发语言·qt·ffmpeg
不惑_11 小时前
用 PyQt5 打造一个可视化 JSON 数据解析工具
开发语言·qt·json
誰能久伴不乏18 小时前
Qt 开发中的父类与父对象的区别和父对象传递:如何选择 `QWidget` 或 `QObject`?
java·开发语言·qt
誰能久伴不乏18 小时前
理解继承与组合的本质:Qt 项目中的设计选择指南
开发语言·qt
抠脚学代码1 天前
Ubuntu18.6 学习QT问题记录以及虚拟机安装Ubuntu后的设置
qt·学习·ubuntu
小道士写程序1 天前
Qt 5.12 上读取 .xlsx 文件(Windows 平台)
开发语言·windows·qt
yxc_inspire2 天前
基于Qt的app开发第十三天
c++·qt·app·tcp·面向对象
潇-xiao2 天前
Qt 按钮类控件(Push Button 与 Radio Button)(1)
c++·qt
追风赶月、2 天前
【QT】认识QT
开发语言·qt