观察者模式详解:用 Qt 信号与槽机制深入理解

引言

你是否曾遇到这样的需求:一个对象的状态发生变化后,希望通知其他对象进行相应的更新?比如:

  • 新闻订阅系统:当新闻发布后,所有订阅者都会收到通知。
  • 股票行情推送:股价变化时,所有关注该股票的投资者都会收到更新信息。
  • Qt 界面开发:按钮被点击时,窗口应该发生某些变化。

这些场景都适用于观察者模式(Observer Pattern)

在本篇文章中,我们不仅讲清楚观察者模式的结构,还会用 Qt 的信号与槽机制 来深入解析,让你真正理解这一模式的奥秘!


1. 什么是观察者模式?

观察者模式是一种 一对多 的设计模式,允许多个对象(观察者)监听某个对象(被观察者)的状态变化,并在变化时收到通知。

简单来说:

  • 被观察者(Subject):负责维护一个观察者列表,并在状态发生变化时通知所有观察者。
  • 观察者(Observer):接收被观察者的通知并做出相应反应。

现实例子:

场景被观察者(Subject)观察者(Observer)
微信公众号订阅公众号订阅的用户
股票市场股票关注股票的投资者
UI 界面按钮点击按钮(QPushButton监听点击的槽函数

2. 观察者模式的结构

观察者模式一般包括以下角色:

  1. Subject(被观察者)

    • 维护一个观察者列表(即谁在关注它)。
    • 当自身状态发生变化时,通知所有观察者。
  2. Observer(观察者)

    • 订阅 Subject,并实现 update() 方法,接收状态变化通知。
  3. 通知机制

    • Subject 需要提供 attach()notify() 方法,管理观察者并进行通知。

观察者模式 UML 结构图

+-------------+       +----------------+
|  Observer   |<------|    Subject     |
+-------------+       +----------------+
| +update()   |       | +attach()      |
|             |       | +detach()      |
|             |       | +notify()      |
+-------------+       +----------------+

3. 传统 C++ 实现观察者模式

在 C++ 中,我们可以用 vector 存储观察者列表,并手动通知它们:

传统 C++ 实现

#include <iostream>
#include <vector>// 观察者接口
class Observer {
public:virtual void update(int value) = 0;
};// 被观察者(Subject)
class Subject {
private:std::vector<Observer*> observers;int state;public:void attach(Observer* observer) { observers.push_back(observer); }void notify() {for (Observer* obs : observers) {obs->update(state);}}void setState(int value) {state = value;notify();}
};// 具体观察者
class ConcreteObserver : public Observer {
public:void update(int value) override {std::cout << "Observer received update: " << value << std::endl;}
};int main() {Subject subject;ConcreteObserver observer1, observer2;subject.attach(&observer1);subject.attach(&observer2);subject.setState(42);  // 触发通知
}

问题:

  • 需要手动管理 Observer 的列表。
  • notify() 需要手动遍历所有观察者,不够灵活。
  • 可能会出现空指针问题(被观察者销毁后,观察者仍在使用)。

4. 用 Qt 信号与槽实现观察者模式

Qt 提供了一种更强大、更安全的实现方式——信号(Signal)与槽(Slot)机制。它本质上就是观察者模式的扩展,但更加灵活和易用!

在这里插入图片描述

信号与槽如何实现观察者模式?

观察者模式角色Qt 信号与槽对应
Subject(被观察者)QObject + Signal
Observer(观察者)QObject + Slot
通知机制connect()

Qt 实现观察者模式

#include <QObject>
#include <QDebug>// 被观察者(Subject)
class Subject : public QObject {Q_OBJECTsignals:void valueChanged(int newValue);  // 信号(事件)public:void setValue(int value) {emit valueChanged(value);  // 触发信号,通知观察者}
};// 观察者(Observer)
class Observer : public QObject {Q_OBJECTpublic slots:void onValueChanged(int newValue) {qDebug() << "Observer received new value:" << newValue;}
};int main() {Subject subject;Observer observer;// 连接信号和槽QObject::connect(&subject, &Subject::valueChanged, &observer, &Observer::onValueChanged);// 触发事件subject.setValue(42);
}

Qt 信号与槽 vs 传统观察者模式

对比项传统观察者模式Qt 信号与槽
观察者管理方式需手动维护列表自动管理
触发方式直接调用 update()emit 发送信号
线程安全需要手动同步Qt::QueuedConnection 支持多线程
解除连接需要手动移除观察者disconnect() 轻松解绑

5. 信号与槽让观察者模式更强大

  1. 自动管理连接,避免空指针错误。
  2. 支持跨线程通信,不需要额外的同步机制。
  3. 更灵活:可以在运行时动态连接和解除连接。
  4. 更易读:代码清晰,不需要手动维护 std::vector<Observer*>

6. 观察者模式如何运用到实际开发中?

1. UI 界面更新

  • 当数据变化时,自动更新 UI 界面(如 QLabelQProgressBar)。
QObject::connect(&subject, &Subject::valueChanged, ui->label, &QLabel::setText);

2. 多线程任务通知主线程

  • 通过 Qt::QueuedConnection 让后台线程通知主线程刷新 UI。

3. 事件驱动开发

  • Qt 本身就是事件驱动的,信号与槽大幅减少了回调函数的使用。

7. 总结

  • 观察者模式解决了 对象间的事件通知 问题。
  • Qt 信号与槽机制 是观察者模式的高级实现,提供自动管理、类型安全、跨线程支持
  • 在 Qt 开发中,几乎所有 UI 组件、后台任务、事件响应都基于 信号与槽

你学会了吗? 🎯 如果有疑问,欢迎讨论! 🚀

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/35687.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

流量分析实践

下载附件使用wireshark打开&#xff0c;发现数据包非常多&#xff0c;一共有1万多条数据&#xff0c;我们点击分析来看一下协议分级 然后我们再来看一下会话&#xff0c;看有哪些ip地址&#xff0c; 我们通过会话结合大部分的流量发现&#xff0c;172.17.0.1一直在请求172.17.0…

新手村:混淆矩阵

新手村&#xff1a;混淆矩阵 一、前置条件 知识点要求学习资源分类模型基础理解分类任务&#xff08;如二分类、多分类&#xff09;和常见分类算法&#xff08;如逻辑回归、决策树&#xff09;。《Hands-On Machine Learning with Scikit-Learn》Python基础熟悉变量、循环、函…

MYSQL库的操作

目录 一、创建数据库 二、字符集和校验规则 1、查看系统默认字符集以及校验规则 2、查看系统支持的所有字符集以及字符集校验规则 3、指定字符集以及校验规则来创建数据库 4、校验规则对数据库的影响 三、操纵数据库 1、查看数据库 2、修改数据库 3、删除数据库 4、数…

Next App Router(下)

五、loading 新增 app/loading.tsx 页面 const Loading () > {return <div>Loading...</div>; }; export default Loading;修改 app/page.tsx页面 /** 假设为一个获取数字的api */ const fetch_getNumber async (): Promise<number> > {return ne…

【JAVA】】深入浅出了解cookie、session、jwt

文章目录 前言一、首先了解http的cookie是什么&#xff1f;Cookie 属性及其含义1. NameValue2. Expires3. Max-Age4. Domain5. Path6. Secure7. HttpOnly8. SameSite示例 Cookie 分类1. Session Cookies2. Persistent Cookies3. First-Party Cookies4. Third-Party Cookies 二、…

【css酷炫效果】纯CSS实现粒子旋转动画

【css酷炫效果】纯CSS实现粒子旋转动画 缘创作背景html结构css样式完整代码效果图 想直接拿走的老板&#xff0c;链接放在这里&#xff1a;https://download.csdn.net/download/u011561335/90492008 缘 创作随缘&#xff0c;不定时更新。 创作背景 刚看到csdn出活动了&…

C++Lambda表达式

Lambda表达式 什么是Lambda表达式 ​ C11的颁布让C丰富了起来&#xff0c;任何一本介绍C11的书籍&#xff0c;都不可能跳过这一个点——Lambda表达式。人们经常称Lambda表达式是一个语法糖&#xff0c;说明这是一个”没有没事&#xff0c;有了更好“的一种语法表达&#xff0…

每天五分钟深度学习框架pytorch:基于pytorch搭建循环神经网络RNN

本文重点 我们前面介绍了循环神经网络RNN,主要分析了它的维度信息,其实它的维度信息是最重要的,一旦我们把维度弄清楚了,一起就很简单了,本文我们正式的来学习一下,如何使用pytorch搭建循环神经网络RNN。 RNN的搭建 在pytorch中我们使用nn.RNN()就可以创建出RNN神经网络…

el-table树形表格合并相同的值

el-table树形表格合并相同的值 el-table树形表格合并相同的值让Ai进行优化后的代码 el-table树形表格合并相同的值 <style lang"scss" scoped> .tableBox {/deep/ &.el-table th:first-child,/deep/ &.el-table td:first-child {padding-left: 0;} } …

2025年3月19日 十二生肖 今日运势

小运播报&#xff1a;2025年3月19日&#xff0c;星期三&#xff0c;农历二月二十 &#xff08;乙巳年己卯月丁亥日&#xff09;&#xff0c;法定工作日。 红榜生肖&#xff1a;兔、虎、羊 需要注意&#xff1a;猪、猴、蛇 喜神方位&#xff1a;正南方 财神方位&#xff1a;…

Git——分布式版本控制工具使用教程

本文主要介绍两种版本控制工具——SVN和Git的概念&#xff0c;接着会讲到Git的安装&#xff0c;Git常用的命令&#xff0c;以及怎么在Vscode中使用Git。帮助新手小白快速上手Git。 1. SVN和Git介绍 1.1 SVN 集中式版本控制工具&#xff0c;版本库是集中存放在中央服务器的&am…

QT5.15.2加载pdf为QGraphicsScene的背景

5.15.2使用pdf 必须要安装QT源码&#xff0c;可以看到编译器lib目录已经有pdf相关的lib文件&#xff0c;d是debug 1.找到源码目录&#xff1a;D:\soft\QT\5.15.2\Src\qtwebengine\include 复制这两个文件夹到编译器的包含目录中:D:\soft\QT\5.15.2\msvc2019_64\include 2.找…

【H2O2 | 软件开发】前端深拷贝的实现

目录 前言 开篇语 准备工作 正文 概述 JSON方法 递归 其他 结束语 前言 开篇语 本系列为短篇&#xff0c;每次讲述少量知识点&#xff0c;无需一次性灌输太多的新知识点。该主题文章主要是围绕前端、全栈开发相关面试常见问题撰写的&#xff0c;希望对诸位有所帮助。…

Docker - 切换源 (Linux / macOS)

文章目录 Linux 系统macOS 系统 Linux 系统 修改配置文件&#xff1a;/etc/docker/daemon.json "registry-mirrors": ["https://docker.mirrors.ustc.edu.cn","https://hub-mirror.c.163.com"]验证是否修改成功&#xff1a; docker info重启 …

hcia复习

一、网络设备 1、交换机&#xff1a;&#xff08;1&#xff09;提供MAC地址表&#xff0c;转发数据&#xff1b; &#xff08;2&#xff09;每个接口是一个独立的冲突域&#xff1b; &#xff08;3&#xff09;凡是连在交换机上的所有设备都处于同一广播域&#xff08;网络&am…

opencv初步学习——图像处理3

这一部分我们将学习opencv中对图像大小进行调整的基本操作&#xff0c;以及掩模操作&#xff0c;我们直接进入正言 一、cv2.resize( )函数 1-1、组成与构造 该函数的作用就算用来帮助我们实现对图像大小的处理&#xff0c;具体的组成与构造如下&#xff1a; cv2.resize(src , …

[LevelDB]关于LevelDB存储架构到底怎么设计的?

本文内容组织形式 LevelDB 存储架构重要特点总体概括LevelDB中内存模型MemTableMemTable的数据结构背景&#xff1a;SkipListSkiplist的数据结构 Skiplist的数据访问细节 SkipList的核心方法Node细节源代码 MemTable的数据加速方式Iterator 的核心方法 MemTable 的读取&写入…

【存储中间件】Redis核心技术与实战(四):Redis高并发高可用(Redis集群 Smart客户端、集群原理)

文章目录 Redis集群Smart客户端smart客户端原理ASK 重定向集群下的Jedis客户端Hash tags 集群原理节点通信通信流程Gossip 消息节点选择 故障转移故障发现主观下线客观下线 故障恢复资格检查准备选举时间发起选举选举投票替换主节点 故障转移时间 集群不可用判定集群读写分离 个…

【接口耗时】⭐️自定义拦截器实现接口耗时统计

&#x1f4a5;&#x1f4a5;✈️✈️欢迎阅读本文章❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;本篇文章阅读大约耗时三分钟。 ⛳️motto&#xff1a;不积跬步、无以千里 &#x1f4cb;&#x1f4cb;&#x1f4cb;本文目录如下&#xff1a;&#x1f381;&#x1f381;&a…

杨校老师课堂之编程入门与软件安装【图文笔记】

亲爱的同学们&#xff0c;热烈欢迎踏入青少年编程的奇妙世界&#xff01; 我是你们的授课老师杨校 &#xff0c;期待与大家一同开启编程之旅。 1. 轻松叩开编程之门 1.1 程序的定义及生活中的应用 程序是人与计算机沟通的工具。在日常生活中&#xff0c;像手机里的各类 APP、电…