设计模式之 观察者模式

观察者模式(Observer Pattern)是一种行为型设计模式,它定义了一种一对多的依赖关系,让多个观察者对象同时监听一个主题对象(Subject)。当主题对象的状态发生变化时,所有依赖于它的观察者都会得到通知并自动更新。观察者模式的应用可以减少对象之间的耦合度,促进系统的解耦,使得对象间的协作变得更加灵活。

观察者模式特别适用于那些需要在一个对象状态变化时,通知多个对象并进行相应处理的场景。它通常用于事件驱动的系统、模型-视图-控制(MVC)架构、GUI框架等应用中。

一、观察者模式的定义

在观察者模式中,主题对象(Subject)保持对多个观察者对象(Observer)的引用,并在自身状态改变时通知它们。观察者对象通过订

阅主题对象来接收通知,而无需主动去查询主题对象的状态。主题和观察者之间通过一个简单的接口进行通信,使得它们之间的耦合度降低。

二、观察者模式的组成部分

观察者模式主要由以下几个角色组成:

  1. Subject(主题/被观察者)

    主题对象是观察者模式的核心,它维护一个观察者列表,并定义向观察者发送通知的方法。主题的状态发生变化时,它会通知所有的观察者。
    常见的操作有:注册观察者、删除观察者、通知观察者。
  2. Observer(观察者)

    观察者是需要接收主题通知的对象,它定义了接收通知的接口。当主题对象状态发生变化时,观察者会自动调用该接口来更新自身状态。
  3. ConcreteSubject(具体主题)

    具体主题类实现了Subject接口,并维护自己的状态。当它的状态变化时,会调用所有注册的观察者的更新方法。
  4. ConcreteObserver(具体观察者)

    具体观察者类实现了Observer接口,并在接收到通知时更新自己的状态。具体观察者与具体主题密切相关,通常它们的状态依赖于主题的状态。

三、观察者模式的工作原理

观察者模式的工作原理可以总结为以下步骤:

  1. 注册观察者:观察者对象通过调用主题对象的attach()方法注册自己,成为主题的观察者。

  2. 通知观察者:当主题的状态发生变化时,主题对象调用notify()方法,通知所有注册的观察者。

  3. 观察者更新:观察者对象收到通知后,会调用自身的update()方法,根据主题的状态更新自己的状态。

  4. 取消订阅:如果观察者不再关心主题的状态变化,它可以通过调用主题的detach()方法注销自己,不再接收通知。

四、观察者模式的代码示例

假设我们有一个简单的天气预报系统,系统需要通知不同的显示设备(如手机、电脑)更新天气信息。当天气变化时,所有已注册的显示设备都会自动接收到天气更新。

定义观察者接口
public interface Observer {public void update(String weather);
}
定义主题接口
public interface Subject {void attach(Observer observer);void detach(Observer observer);void notifyObservers();
}
具体主题类:WeatherStation(天气站)
public class WeatherStation implements Subject{private List<Observer> observers = new ArrayList<>();private String weather;public void setWeather(String weather) {this.weather = weather;notifyObservers();}@Overridepublic void attach(Observer observer) {observers.add(observer);}@Overridepublic void detach(Observer observer) {observers.remove(observer);}@Overridepublic void notifyObservers() {for (Observer observer : observers) {observer.update(weather);}}
}
具体观察者类:PhoneDisplay(手机显示)
public class PhoneDisplay implements Observer{@Overridepublic void update(String weather) {System.out.println("手机端天气变为"+ weather);}
}
具体观察者类:ComputerDisplay(电脑显示)
public class ComputerDisplay implements Observer{@Overridepublic void update(String weather) {System.out.println("电脑端天气变为"+weather);}
}
客户端代码
public class Client {public static void main(String[] args) {WeatherStation station = new WeatherStation();station.attach(new ComputerDisplay());station.attach(new PhoneDisplay());station.setWeather("晴天");}
}
运行结果

五、观察者模式的优缺点

优点:
  1. 降低耦合性:观察者模式使得主题和观察者之间的耦合度降低,观察者无需了解主题的内部实现,主题也无需了解每个观察者的具体细节。只通过接口进行通信。

  2. 支持多种观察者:多个观察者可以同时订阅同一个主题对象,在主题对象状态发生变化时,所有观察者都会被通知并做出响应。

  3. 增强灵活性:如果主题的状态发生变化,观察者无需主动查询主题,而是通过通知机制自动获取更新。观察者可以随时添加或移除,不影响系统其他部分。

  4. 扩展性强:如果需要新增新的观察者,只需要创建新的观察者类,并注册到主题中即可,主题对象不需要做任何修改,符合开放封闭原则。

缺点:
  1. 可能引起性能问题:如果观察者非常多,每当主题状态变化时,都会通知所有观察者。如果观察者处理较慢或者状态更新频繁,可能会导致性能问题。

  2. 观察者过多时管理困难:如果观察者的数量非常大,管理这些观察者可能变得复杂,特别是在需要追踪哪些观察者已经收到通知时。

  3. 难以控制通知的顺序:在一些复杂的场景中,可能希望通知观察者的顺序有一定的控制,而观察者模式通常是按照注册顺序发送通知的。

六、观察者模式的应用场景

  1. 事件处理系统:在事件驱动的系统中,事件的发生通常需要通知多个监听者,如图形界面中的按钮点击、鼠标拖拽等事件。

  2. 模型-视图-控制(MVC)模式:在MVC架构中,视图(Observer)需要根据模型(Subject)的变化来更新显示。模型的状态变化会通知所有相关的视图进行刷新。

  3. 通知推送系统:当数据变化时,通知用户的系统,比如邮件通知、即时通讯应用中的消息推送等。

  4. 股票市场监控:投资者或分析师(观察者)订阅股票(主题)信息,当股票价格发生变化时,系统会自动推送通知给所有订阅者。

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

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

相关文章

深入了解 Linux htop 命令:功能、用法与示例

文章目录 深入了解 Linux htop 命令&#xff1a;功能、用法与示例什么是 htop&#xff1f;htop 的安装htop的基本功能A区&#xff1a;系统资源使用情况B区&#xff1a;系统概览信息C区&#xff1a;进程列表D区&#xff1a;功能键快捷方式 与 top 的对比常见用法与示例实际场景应…

【深度学习】【RKNN】【C++】模型转化、环境搭建以及模型部署的详细教程

【深度学习】【RKNN】【C】模型转化、环境搭建以及模型部署的详细教程 提示:博主取舍了很多大佬的博文并亲测有效,分享笔记邀大家共同学习讨论 文章目录 【深度学习】【RKNN】【C】模型转化、环境搭建以及模型部署的详细教程前言模型转换--pytorch转rknnpytorch转onnxonnx转rkn…

【spark】远程debug spark任务(含有pyspark)

--master yarn和--master client都是可以的。 spark-submit \ --master yarn \ --deploy-mode client \ --name "test-remote-debug" \ --conf "spark.driver.extraJavaOptions-agentlib:jdwptransportdt_socket,servery,suspendn,address5005" \ --conf …

如何使用 Vivado 从源码构建 Infinite-ISP FPGA 项目

如约介绍源码构建 Infinite-ISP 项目&#xff0c;其实大家等的是源码&#xff0c;所以中间过程简洁略过&#xff0c;可以直接翻到文末获取链接。 开源ISP&#xff08;Infinite-ISP&#xff09;介绍 构建工程 第一步&#xff0c;从文末或者下面链接获取源码 https://github.com/…

彻底理解Redis的持久化方式

一.由来 因为Redis之所以能够提供高效读写的操作&#xff0c;是因为它是基于内存的&#xff0c;但是这样也会带来一个问题&#xff0c;及在服务器宕机或者重启的情况下&#xff0c;内存里面的数据就会被丢失掉&#xff0c;所以为了解决这个问题&#xff0c;Redis就提供了持久化…

Bug Fix 20241122:缺少lib文件错误

今天有朋友提醒才突然发现 gitee 上传的代码存在两个很严重&#xff0c;同时也很低级的错误。 因为gitee的默认设置不允许二进制文件的提交&#xff0c; 所以PH47框架下的库文件&#xff08;各逻辑层的库文件&#xff09;&#xff0c;以及Stm32Cube驱动的库文件都没上传到Gi…

NVR管理平台EasyNVR多个NVR同时管理:全方位安防监控视频融合云平台方案

EasyNVR是基于端-边-云一体化架构的安防监控视频融合云平台&#xff0c;具有简单轻量的部署方式与多样的功能&#xff0c;支持多种协议&#xff08;如GB28181、RTSP、Onvif、RTMP&#xff09;和设备类型&#xff08;IPC、NVR等&#xff09;&#xff0c;提供视频直播、录像、回放…

微服务架构:10个实用设计模式

1 微服务架构 微服务架构的重要特征 微服务架构的优点 微服务架构的缺点 何时使用微服务架构 2 微服务架构的设计模式 独享数据库&#xff08;Database per Microservice&#xff09; 事件源&#xff08;Event Sourcing&#xff09; 命令和查询职责分离&#xff08;CQRS&…

华为欧拉系统使用U盘制作引导安装华为欧拉操作系统

今天记录一下通过U盘来安装华为欧拉操作系统 华为欧拉操作系统是国产的一个类似于Centos的Linus系统 具体实现操作步骤&#xff1a; 先在官网下载欧拉系统镜像点击跳转到下载 准备好一个大于16g的U盘 &#xff0c;用于制作U盘启动 下载一个引导程序制作工具&#xff0c;我使用…

20241121 android中树结构列表(使用recyclerView实现)

1、adapter-item的布局 <?xml version"1.0" encoding"utf-8"?> <LinearLayout xmlns:android"http://schemas.android.com/apk/res/android"android:layout_width"match_parent"android:layout_height"wrap_content&…

C++初阶学习 第十二弹——stack与queue的介绍和使用

目录 一、stack&#xff08;栈&#xff09; 1.栈的概念&#xff1a; 2.成员函数包括&#xff1a; 3.栈的使用示例: 4. 使用时的注意事项&#xff1a; 二.queue&#xff08;队列&#xff09; 1.队列的概念 2.成员函数 3.队列的使用示例 4.使用时的注意事项 三.总结…

如何实现点击目录跳转到指定位置?【vue】

需求&#xff1a;实现目录点击跳转到指定位置&#xff0c;点击后直接定位到指定模块 效果&#xff1a; 实现方法&#xff1a; &#xff08;1&#xff09;a标签跳转 普通使用&#xff1a; <!DOCTYPE html> <html><head><title>a-Demo</title>&l…

【SKFramework框架】二、快速启动

推荐阅读 CSDN主页GitHub开源地址Unity3D插件分享QQ群&#xff1a;398291828小红书小破站 大家好&#xff0c;我是佛系工程师☆恬静的小魔龙☆&#xff0c;不定时更新Unity开发技巧&#xff0c;觉得有用记得一键三连哦。 一、前言 【Unity3D框架】SKFramework框架完全教程《全…

鸿蒙多线程开发——线程间数据通信对象02

1、前 言 本文的讨论是接续鸿蒙多线程开发——线程间数据通信对象01的讨论。在上一篇文章中&#xff0c;我们讨论了常规的JS对象(普通JSON对象、Object、Map、Array等)、ArrayBuffer。其中讨论了ArrayBuffer的复制传输和转移传输方式。 下面&#xff0c;我们将讨论SharedArra…

时序论文22|ICML24港科大:面向多变量不规则的时间序列预测方法

论文标题&#xff1a;Irregular Multivariate Time Series Forecasting: A Transformable Patching Graph Neural Networks Approach 论文链接&#xff1a;https://openreview.net/pdf?idUZlMXUGI6e 前言 这篇文章在“定位研究问题”方面很值得学习。其实前段时间对时序研究…

Linux离线安装Docker命令,简单镜像操作

解压安装包 首先&#xff0c;使用 tar 命令解压 docker-27.3.1.tgz 安装包&#xff1a; tar -zxvf docker-27.3.1.tgz 将二进制文件移动到可执行路径上的目录 接着&#xff0c;将解压出来的 Docker 二进制文件复制到系统的可执行路径&#xff08;通常是 /usr/bin/&#xff09…

Sigrity SPEED2000 TDR TDT Simulation模式如何进行时域阻抗仿真分析操作指导-差分信号

Sigrity SPEED2000 TDR TDT Simulation模式如何进行时域阻抗仿真分析操作指导-差分信号 Sigrity SPEED2000 TDR TDT Simulation模式如何进行时域阻抗仿真分析操作指导-单端信号详细介绍了单端信号如何进行TDR仿真分析,下面介绍如何对差分信号进行TDR分析,还是以下图为例进行分…

springboot基于微信小程序的食堂预约点餐系统

摘 要 基于微信小程序的食堂预约点餐系统是一种服务于学校和企事业单位食堂的智能化解决方案&#xff0c;旨在提高食堂就餐的效率、缓解排队压力&#xff0c;并优化用户的就餐体验。系统作为一种现代化的解决方案&#xff0c;为食堂管理和用户就餐提供了便捷高效的途径。它不仅…

Java线程池详解

线程池是用来管理和复用线程的一种技术&#xff0c;它避免了频繁的创建和销毁线程的开销&#xff0c;提高了应用程序的性能。在 Java 中&#xff0c;ExecutorService 是一个非常常用的接口&#xff0c;它提供了线程池的基本功能。 1. 线程池的优势 线程复用&#xff1a;线程池…

软件测试—— Selenium 常用函数(二)

前一篇文章&#xff1a;软件测试—— Selenium 常用函数&#xff08;一&#xff09;-CSDN博客 目录 前言 一、浏览器 1.常见操作 &#xff08;1&#xff09;打开网站 &#xff08;2&#xff09;前进、后退、刷新 2.参数设置 &#xff08;1&#xff09;设置无头模式 &am…