技术成神之路:设计模式(九)观察者模式

介绍

观察者模式(Observer Pattern)是一种行为设计模式。它允许一个对象(称为主题或可观察者)来监视并通知一组依赖于这个对象的其他对象(称为观察者),以便在主题状态发生变化时自动更新观察者的对象。

1.定义


观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。这个主题对象在状态变化时,会通知所有注册过的观察者对象,使它们能够自动更新自己。

2.主要作用


主要作用是解耦主题与观察者之间的耦合关系,使得主题和观察者之间可以轻松地进行交互和通信。

3.解决的问题


  • 对象间的依赖关系管理:将对象间的依赖关系从静态写死的调用关系解耦,通过抽象出观察者接口和主题接口来管理对象间的关系。

  • 通知机制:当一个对象的状态发生变化时,需要通知其他相关的对象做出相应的改变,观察者模式提供了一种灵活的通知机制。

观察者模式应该是我们在开发中使用最频繁的一个设计模式了,比如在安卓中经常用到的:

  1. LiveData 和 Observer:LiveData 是一种在 Android 应用中生命周期感知的可观察数据持有类。可以存储数据,并在数据发生变化时通知其观察者。通常与 Observer 结合使用。
  2. BroadcastReceiver 和 IntentFilter:广播发送者和接收者之间的通信机制可以看作是一种观察者模式的实现。
  3. RxJava 和 Observer:提供了 Observables 和 Observers 的概念,可以用来实现观察者模式。

注意 观察者模式的定义是一对多的关系,像监听事件只能算回调(Callback),不严格符合传统的观察者模式的定义!

4.模式原理


包含角色:

  • Subject(主题):维护一组观察者对象,提供注册和删除观察者对象的接口。主题对象状态发生变化时,调用观察者的相应方法。

  • Observer(观察者):定义一个更新接口,使得在主题对象状态改变时能够得到通知和更新。

  • ConcreteSubject(具体主题):将有关状态存入具体观察者对象;在具体主题的内部状态改变时,给所有登记过的观察者发出通知。

  • ConcreteObserver(具体观察者):维护一个指向具体主题对象的引用,存储与主题的状态相对应的信息。实现 Observer 定义的更新接口,以便使自身状态与主题的状态保持一致。

UML类图:
在这里插入图片描述
示例代码:

主题接口 Subject

import java.util.ArrayList;
import java.util.List;// 主题接口
interface Subject {void registerObserver(Observer observer);void removeObserver(Observer observer);void notifyObservers();
}// 具体主题类
class ConcreteSubject implements Subject {private List<Observer> observers = new ArrayList<>();private int state;public int getState() {return state;}public void setState(int state) {this.state = state;notifyObservers();}@Overridepublic void registerObserver(Observer observer) {observers.add(observer);}@Overridepublic void removeObserver(Observer observer) {observers.remove(observer);}@Overridepublic void notifyObservers() {for (Observer observer : observers) {observer.update(state);}}
}

观察者接口 Observer

// 观察者接口
interface Observer {void update(int state);
}// 具体观察者类
class ConcreteObserver implements Observer {private int observerState;@Overridepublic void update(int state) {observerState = state;System.out.println("Observer state updated: " + observerState);}
}

使用

public class ObserverPatternExample {public static void main(String[] args) {// 创建具体主题对象ConcreteSubject subject = new ConcreteSubject();// 创建具体观察者对象ConcreteObserver observer1 = new ConcreteObserver();ConcreteObserver observer2 = new ConcreteObserver();// 注册观察者subject.registerObserver(observer1);subject.registerObserver(observer2);// 改变主题状态,并触发通知subject.setState(10);// 移除一个观察者subject.removeObserver(observer1);// 再次改变主题状态,并触发通知subject.setState(20);}
}

打印

Observer state updated: 10
Observer state updated: 10Observer state updated: 20

上面举例为标准的观察者模式写法,我记得在上家公司代码中 有一种奇怪的写法,当然,也符合观察者模式定义。

在安卓中不同组件之间大多都有传递消息的需求,大家常用的库有EventBus吧。简单,方便,它是定义了一个 IHandler类,其中代码 就像上面 ConcreteSubject 类中差不多,只不过多了一个Handler 就是为了方便切换线程的,保证观察者收到的消息都是在主线程中,注意他是一个类,不是接口,而且所有代码都在IHandler 类中,谁继承它 就是 具体主题类 ,因为安卓是单继承嘛,这个使用起来很不方便。

无论你在任何地方,想要得到这个 具体主题类的消息,只需要向它注册观察者即可,当然最后别忘了反注册,不然要内存泄漏了,刚用的时候稍不留意就忘了,可恶心,当然 这个具体主题类 一般都是单例,全局唯一对象,因为当时是霸屏应用(特殊场景),所以单例满天飞,用的时候可爽,哈哈哈,现在回想起来都是问题。

5.优缺点


优点:

  • 解耦性:主题和观察者之间是抽象耦合,容易扩展。
  • 支持广播通信:主题可以通知多个观察者,观察者也可以接收多个主题的通知。
  • 灵活性:可以动态地设置通知方式和观察者的添加和删除。

缺点:

  • 可能引起循环调用:如果观察者和主题之间的依赖关系设计不当,可能导致循环调用。
  • 通知效率:如果通知过于频繁,可能会影响系统性能,一般场景这个问题不用考虑的,但如果你的观察者好多个甚至几十上百的话 就不得不考虑效率问题了。

6.应用场景


  1. 当一个抽象模型有两个方面,其中一个依赖于另一个。
  2. 当一个对象的改变需要同时改变其他对象,而不知道具体有多少对象需要改变时。
  3. 当一个对象必须通知其他对象,而它又不能假定其他对象是谁。

7.总结


观察者模式通过定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象,实现了对象之间的解耦和通信。它是实现事件驱动程序设计的基础,广泛应用于GUI(Graphical User Interface,图形用户界面)开发、事件处理系统等需要高内聚、低耦合的场景中。

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

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

相关文章

Python中高效处理大数据的几种方法

随着数据量的爆炸性增长&#xff0c;如何在Python中高效地处理大数据成为了许多开发者和数据科学家的关注焦点。Python以其简洁的语法和丰富的库支持&#xff0c;在数据处理领域占据了重要地位。本文将介绍几种在Python中高效处理大数据的常用方法。 目录 1. 使用Pandas进行数…

双向链表(C语言版)

1. 双向链表的结构 注意&#xff1a;这里的“带头”跟单链表的“头结点”是两个概念&#xff0c;实际上在单链表阶段称呼不太严谨&#xff0c;但是为了更好地理解就直接称为单链表的头结点。带头链表里的头结点&#xff0c;实际为“哨兵位”&#xff0c;哨兵位结点不存储任何有…

若依Vue前后端分离版如何部署(windows)(超详细)

一、项目环境准备 下面是项目所需要准备的环境 Node.js redis 1、Node.js下载 下面进入官网可以下载Node.js — 在任何地方运行 JavaScript (nodejs.org)https://nodejs.org/zh-cn 下载完成安装后&#xff0c;需要配置环境变量&#xff0c;首先复制以下nodejs的安…

【JavaScript】箭头函数

具体讲解 之前写 this 的指向时就提到过箭头函数&#xff0c;但是由于其比较复杂&#xff0c;还是单独开一篇来讲箭头函数。 箭头函数&#xff0c;箭头函数不能作为构造函数&#xff0c;没有原型 prototype&#xff0c;不能 new。 在箭头函数中&#xff0c;this 关键字指向的是…

代码随想录Day15|| 110平衡二叉树 257二叉树的所有路径 404左叶子之和 222完全二叉树的节点个数

110平衡二叉树 力扣题目链接 题目描述 给定一个二叉树&#xff0c;判断它是否是 平衡二叉树 示例 1&#xff1a; 输入&#xff1a;root [3,9,20,null,null,15,7] 输出&#xff1a;true示例 2&#xff1a; 输入&#xff1a;root [1,2,2,3,3,null,null,4,4] 输出&#xff…

.NET Core异步编程与多线程解析:提升性能与响应能力的关键技术

在.NET Core中&#xff0c;异步编程和多线程是构建高性能应用程序的核心技能。理解这两个概念不仅可以提升应用程序的响应能力&#xff0c;还能优化资源使用。本文将深入剖析异步编程和多线程的关键知识点&#xff0c;提供代码示例&#xff0c;并附上步骤以帮助理解。 1. 异步…

深入浅出WebRTC—GCC

GoogCcNetworkController 是 GCC 的控制中心&#xff0c;它由 RtpTransportControllerSend 通过定时器和 TransportFeedback 来驱动。GoogCcNetworkController 不断更新内部各个组件的状态&#xff0c;并协调组件之间相互配合&#xff0c;向外输出目标码率等重要参数&#xff0…

高级java每日一道面试题-2024年7月24日-你对ReadWriteLock和StampedLock了解多少?

面试官: 你对ConcurrentHashMap了解多少? 我回答: ReadWriteLock和StampedLock都是Java并发库中提供的锁机制&#xff0c;它们各自针对不同场景提供了灵活性和性能优势。 ReadWriteLock ReadWriteLock是Java并发包中的一个接口&#xff0c;它提供了一种读写锁的实现&#…

昇思25天学习打卡营第17天|计算机视觉

昇思25天学习打卡营第17天 文章目录 昇思25天学习打卡营第17天ShuffleNet图像分类ShuffleNet网络介绍模型架构Pointwise Group ConvolutionChannel ShuffleShuffleNet模块构建ShuffleNet网络 模型训练和评估训练集准备与加载模型训练模型评估模型预测 打卡记录 ShuffleNet图像分…

2023 N1CTF-n1proxy

文章目录 参考rsa握手rust_proxy源码公匙交换和签名会话钥匙后续通信生命周期和裸指针代码审计漏洞点 libc-2.27.so大致思路&#xff08;exp还有变化&#xff09;调试exp泄露libc写free_hook执行命令exp 参考 https://github.com/Nu1LCTF/n1ctf-2023/tree/main/pwn/n1proxy ht…

JVM从1%到99%【精选】-运行时数据区

目录 1.运行时数据区概括 2.什么是内存溢出 3..程序计数器 4.Java虚拟机栈 5.本地方法栈 6.堆 7.方法区 8.直接内存 1.运行时数据区概括 Java虚拟机在运行Java程序过程中管理的内存区域,称之为运行时数据区。主要分为两大类&#xff1a;线程不共享、线程共享线程不共…

14 B端产品的运营管理

通过运营找到需求并通过交换价值提供供给&#xff0c;再逐步扩大规模、站稳脚跟&#xff0c;辅助产品在商业竞争中获胜。 B端产品运营框架 1. 打通渠道 目的&#xff1a;触达客户。 环节&#xff1a;文案策划、活动策划→广告渠道推广→线下BD。 线下BD&#xff1a;通过见面…

2024华为OD机试真题- 亲子游戏-(C++/Java/Python)-C卷D卷-200分

2024华为OD机试题库-(C卷+D卷)-(JAVA、Python、C++) 题目描述 宝宝和妈妈参加亲子游戏,在一个二维矩阵(N*N)的格子地图上,宝宝和妈妈抽签决定各自的位置,地图上每个格子有不同的糖果数量,部分格子有障碍物。 游戏规则是妈妈必须在最短的时间(每个单位时间只能走一步)…

TypeScript中Interface接口的深度探索与实践

定义接口 在TypeScript中&#xff0c;interface是一个强有力的概念&#xff0c;它用于定义类型签名&#xff0c;特别是对象的结构。接口可以用来描述对象应该有哪些属性、方法&#xff0c;以及这些成员的类型。它们是实现类型系统中“鸭子类型”&#xff08;duck typing&#…

typescript中type和interface有什么区别,如何选择?

定义&#xff1a; interface Users {name: string;age: number;getName: () > string; } type UsersType {name: string;age: number;getName: () > string; }使用&#xff1a; const user: Users {name: 张三,age: 18,getName() {return this.name} }class Person i…

Python基础知识—一文了解numpy

目录 导入和使用 多维数组 多维数组的基本操作 数组的算数运算 数组的自身运算 随机数组 导入和使用 安装完成后&#xff0c;可以在Python中使用以下代码来验证NumPy是否已经正确安装&#xff1a; import numpy as np a np.array([1, 2, 3])print(a) 多维数组 ⚪️ …

【22】Android高级知识之Window(三) -WMS

一、概述 这次开始到了WindowManagerService&#xff08;WMS&#xff09;&#xff0c;你可以把它看做一个WindowManager&#xff0c;只不过呢&#xff0c;属于系统服务进程&#xff08;system_server&#xff09;中的一员&#xff0c;和应用不在同一进程&#xff0c;所以涉及了…

CSS(二)——CSS 背景

CSS 背景 CSS 背景属性用于定义HTML元素的背景。 CSS 背景属性 Property描述background简写属性&#xff0c;作用是将背景属性设置在一个声明中。background-attachment背景图像是否固定或者随着页面的其余部分滚动。background-color设置元素的背景颜色。background-image把…

C# 设计模式分类

栏目总目录 1. 创建型模式&#xff08;Creational Patterns&#xff09; 创建型模式主要关注对象的创建过程&#xff0c;包括如何实例化对象&#xff0c;并隐藏实例化的细节。 单例模式&#xff08;Singleton&#xff09;&#xff1a;确保一个类只有一个实例&#xff0c;并提…

go使用gjson操作json数据

gjson使用 gjson介绍安装gjson库解析 JSON 字符串路径语法复杂查询遍历和修改结论 gjson介绍 gjson 是一个 Go 语言库&#xff0c;用于快速解析 JSON 数据。它提供了非常简洁的 API 来查询 JSON 数据&#xff0c;无需预先定义 Go 结构体或映射&#xff08;map&#xff09;来匹…