当前位置: 首页 > news >正文

观察者设计模式知多少

目录

目标

概述

实现 

推设计模式

拉设计模式

被动观察者设计模式


目标

熟悉观察者设计模式,了解观察者设计模式的使用场景、具体实现(包括:推设计模式、拉设计模式、被动观察者设计模式)


概述

一、行为设计模式

行为设计模式是设计模式的一种类型。该类型的设计模式关注的重点在于对象的行为(通信和交互),而非对象的创建方式。较为常见的行为设计模式有:观察者模式、策略模式、命令模式、责任链模式等。优点是减少了对象之间的耦合度

二、观察者设计模式

对象之间存在一对多的依赖关系,多个观察者对象同时监听某一个主题对象(被观察者对象)。当主题对象的状态发生改变时会通知其他观察者对象。主题对象实现思路:

  1. 主题对象类定义一个集合observerList用来保存多个观察者对象。
  2. 定义观察者对象注册到主题对象的方法、主题对象从注册表中剔除指定观察者对象的方法、定义通知方法。
  3. 当对象状态发生改变时调用该方法,以遍历observerList的方式通知所有观察者对象。

优点

  1. 被观察者使用注册表的方式囊括了所有观察者,降低了观察者对象和被观察者对象之间的依赖关系,提高了系统的扩展性和可维护性。
  2. 被观察者类提供了注册,剔除的方法,使得代码扩展性增强。
  3. 该模式实现了事件驱动的方式,使业务实现方案变得更灵活。

缺点

  1. 观察者过多会导致内存开销大。
  2. 容易发生循环依赖问题。

种类

  • 推设计模式:最简单的模式,指被观察者主动将状态推送给观察者。
  • 拉设计模式:被观察者只告知观察者状态发生了改变,观察者需要通过主动去“拉”的方式,获取被观察者的具体状态。
  • 被动观察者设计模式:引入了一个中介者对象来协调主题对象和观察者之间的交互。主题对象不再直接通知观察者,而是将状态变化通知给中介者对象,由中介者对象负责通知观察者。

三、使用场景

  • 一个对象发生改变需要通知其他对象,其他对象可以自由选择监听或者不监听。
  • 一个对象发生改变需要通知其他对象,其他对象的类型和数量不确定时。

四、列举观察者模式在成熟的框架中的应用

  • 在RabbitMQ中,Exchange和Queue之间的绑定关系。
  • Java AWT/Swing事件处理机制。
  • Spring Framework事件机制。
  • Java Message Service(JMS)。

实现 

推设计模式

需求

多个用户订阅了天气预报系统,则天气预报作为被观察者对象,一旦天气预报发生更新,则订阅者需要马上感知并获取天气预报状态。

观察者接口

package com.ctx.observer.push;// 定义用户作为观察者对象
interface Observer {public void update(String weather);
}

观察者接口实现类

package com.ctx.observer.push;// 实现具体的观察者对象
class User implements Observer {private String name;public User(String name) {this.name = name;}// 接收天气预报信息public void update(String weather) {System.out.println(name + "收到天气预报信息:" + weather);}
}

被观察者类

package com.ctx.observer.push;import java.util.ArrayList;
import java.util.List;// 定义天气预报机构作为主题对象
class WeatherForecast {private List<Observer> observers = new ArrayList<Observer>();private String weather;// 注册观察者对象public void registerObserver(Observer observer) {observers.add(observer);}// 移除观察者对象public void removeObserver(Observer observer) {observers.remove(observer);}// 通知所有观察者对象public void notifyObservers() {for (Observer observer : observers) {observer.update(weather);}}// 更新天气预报信息public void setWeather(String weather) {this.weather = weather;notifyObservers();}
}

调用测试类

package com.ctx.observer.push;// 测试代码
public class Test {public static void main(String[] args) {WeatherForecast weatherForecast = new WeatherForecast();// 创建多个用户作为观察者对象User user1 = new User("张三");User user2 = new User("李四");User user3 = new User("王五");// 注册观察者对象weatherForecast.registerObserver(user1);weatherForecast.registerObserver(user2);weatherForecast.registerObserver(user3);// 发布天气预报信息weatherForecast.setWeather("今天晴朗,温度26℃");// 移除观察者对象weatherForecast.removeObserver(user2);// 发布天气预报信息weatherForecast.setWeather("明天有雨,温度22℃");}
}

拉设计模式

需求

多个用户订阅了新闻频道,则新闻频道作为被观察者对象,一旦新闻频道发生更新,则订阅者需要马上感知新闻频道发生了更新。但是具体更新了什么新闻,需要由用户主动去获取。

观察者接口

package com.ctx.observer.pull;// 定义观察者对象
interface Observer {void update(News news);
}

观察者接口实现类

package com.ctx.observer.pull;// 定义具体的观察者对象(居民对象)
class Resident implements Observer {private String name;public Resident(String name) {this.name = name;}// 拉模式:观察者主动获取消息内容public void update(News news) {System.out.println(name + "收到消息: " + news.getNewsContent());}public String getName() {return name;}
}

被观察者类

package com.ctx.observer.pull;import java.util.ArrayList;
import java.util.List;// 定义被观察者对象(新闻对象是被观察者)
class News {private List<Observer> observers = new ArrayList<>();private String newsContent;public void addObserver(Observer observer) {observers.add(observer);}public void removeObserver(Observer observer) {observers.remove(observer);}public void setNewsContent(String newsContent) {this.newsContent = newsContent;notifyObservers();}// 通知观察者更新消息private void notifyObservers() {for (Observer observer : observers) {observer.update(this);}}// 获取消息内容public String getNewsContent() {return newsContent;}
}

调用测试类

package com.ctx.observer.pull;public class Test {public static void main(String[] args) {News news = new News();// 居民订阅新闻Resident resident1 = new Resident("张三");Resident resident2 = new Resident("李四");news.addObserver(resident1);news.addObserver(resident2);// 设置新闻内容,观察者拉取消息内容news.setNewsContent("今天是个好日子!");}
}

被动观察者设计模式

需求

患者住院治疗,则患者作为被观察者,医生和护士作为观察者。但是医生和护士不能一直陪在患者身边,为了及时响应,医院安装了医疗监护系统,一旦患者有需要则通过医疗监护系统通知医生和护士,医疗监护系统起到了中介者的作用,它来协调观察者和被观察者之间的交互。

观察者接口

package com.ctx.observer.passive;// 定义观察者接口
interface Observer {void update(); // 更新观察者状态
}

观察者接口实现类

package com.ctx.observer.passive;// 定义具体的观察者,即医生和护士类
class MedicalStaff implements Observer {private String name;private Subject patient;public MedicalStaff(String name, Subject patient) {this.name = name;this.patient = patient;patient.registerObserver(this); // 注册观察者}@Overridepublic void update() {String condition = ((Patient) patient).getCondition(); // 获取被观察者状态System.out.println(name + "收到病人病情更新通知,病情为:" + condition); // 处理病情变化}
}

中介者

package com.ctx.observer.passive;// 定义中介者,即医疗监护系统类
class MedicalMonitor {private Subject patient;public MedicalMonitor(Subject patient) {this.patient = patient;}public void setPatientCondition(String condition) {((Patient) patient).setCondition(condition); // 更新被观察者状态}
}

被观察者接口

package com.ctx.observer.passive;// 定义被观察者接口
interface Subject {void registerObserver(Observer observer); // 注册观察者void removeObserver(Observer observer); // 移除观察者void notifyObservers(); // 通知观察者
}

被观察者接口实现类

package com.ctx.observer.passive;import java.util.ArrayList;
import java.util.List;// 定义具体的被观察者,即病人类
class Patient implements Subject {private List<Observer> observers; // 观察者列表private String condition; // 病情public Patient() {observers = new ArrayList<>();}@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(); // 通知每个观察者更新状态}}public void setCondition(String condition) {this.condition = condition;notifyObservers(); // 状态变化后通知所有观察者更新状态}public String getCondition() {return condition;}
}

调用测试类

package com.ctx.observer.passive;// 测试
public class Test {public static void main(String[] args) {Patient patient = new Patient(); // 创建被观察者MedicalMonitor monitor = new MedicalMonitor(patient); // 创建中介者MedicalStaff doctor = new MedicalStaff("张医生", patient); // 创建观察者MedicalStaff nurse = new MedicalStaff("李护士", patient); // 创建观察者monitor.setPatientCondition("发热"); // 病人状态变化,触发通知patient.removeObserver(doctor); // 病人移除一个观System.out.println("--------------------------");monitor.setPatientCondition("感冒"); // 病人状态变化,触发通知}
}

相关文章:

观察者设计模式知多少

目录 目标 概述 实现 推设计模式 拉设计模式 被动观察者设计模式 目标 熟悉观察者设计模式&#xff0c;了解观察者设计模式的使用场景、具体实现&#xff08;包括&#xff1a;推设计模式、拉设计模式、被动观察者设计模式&#xff09;。 概述 一、行为设计模式 行为设…...

Flink之TaskManager内存解析

一、CK失败 Flink任务的checkpoint操作失败大致分为两种情况&#xff0c;ck decline和ck expire: &#xff08;1&#xff09;ck decline 发生ck decline情况时&#xff0c;我们可以通过查看JobManager.log或TaskManager.log查明具体原因。其中有一种特殊情况为ck cancel&…...

为何越来越多人不喜欢“试用期六个月”的公司?网友:感觉不靠谱

众所周知&#xff0c;任何一份工作都有试用期&#xff0c;一般是三月左右。但如果你遇到试用期达到半年的公司&#xff0c;你会不会进入&#xff1f; 近日&#xff0c;就有人遇到了此类公司&#xff0c;并对是否要进入该公司犹豫不决。他在论坛上发帖求助&#xff1a;大家是怎…...

单例模式的四种创建方式

前言 单例模式是日常开发中最常见的一种设计模式&#xff0c;常用来做为池对象&#xff0c;或者计数器之类的需要保证全局唯一的场景。 单例模式的目的是保证在整个程序中只存在一个对象实例&#xff0c;使用单例一个前提条件就是构造器私有化&#xff0c;不允许通过new 对象…...

Nginx+Keepalived 中的脑裂现象

如何解决和预防 NginxKeepalived 中会出现的脑裂现象? Nginx是一种高性能的Web服务器和反向代理服务器&#xff0c;可以处理大量并发请求。Keepalived是一种开源软件&#xff0c;用于实现IP负载均衡和故障转移。在Nginx和Keepalived结合使用时&#xff0c;可以通过将多个Ngin…...

04 KVM虚拟化网络概述

文章目录 04 KVM虚拟化网络概述4.1 Linux Bridge4.2 Open vSwitch 04 KVM虚拟化网络概述 为了使虚拟机可以与外部进行网络通信&#xff0c;需要为虚拟机配置网络环境。KVM虚拟化支持Linux Bridge、Open vSwitch网桥等多种类型的网桥。如图1所示&#xff0c;数据传输路径为“虚…...

110页智慧农业解决方案(农业信息化解决方案)(ppt可编辑)

本资料来源公开网络&#xff0c;仅供个人学习&#xff0c;请勿商用&#xff0c;如有侵权请联系删除。 第一部分 智慧农业概述 智慧农业以农业资源为基础、市场为导向、效益为中心、产业化为抓手&#xff0c;面向农业管理部门、农技推广部门、农业企业、农业园区和基地、农业专…...

Java知识体系及聊天室程序

Java知识体系结构梳理如下&#xff1a; 基础语法&#xff1a;Java的基本语法&#xff0c;包括数据类型、运算符、控制语句、数组等。 面向对象编程&#xff1a;Java是一种面向对象的编程语言&#xff0c;需要掌握类、对象、继承、多态等概念。 异常处理&#xff1a;Java提供了…...

java的详细发展历程

Java是一种跨平台、面向对象的编程语言&#xff0c;具有简单性、可移植性、安全性等特点。Java的历史可以追溯到上世纪90年代初期&#xff0c;以下是Java的详细发展历程&#xff1a; 1991年&#xff0c;Sun Microsystems公司的James Gosling和他的团队开始开发一种名为Oak的编程…...

丢石子

I 一堆石子有n个,两人轮流取.先取者第1次可以取任意多个&#xff0c;但不能全部取完.以后每次取的石子数不能超过上次取子数的2倍。取完者胜.先取者负输出"Second win".先取者胜输出"First win". 思路&#xff1a; 任何正整数都可以表示为不连续斐波那契…...

skywalking手动上报一些指标信息

skywalking的相关概念我就不介绍了&#xff0c;有兴趣可以参看官网文档 以下提供以下简单示例手工上报一些对问题排查比较有用的一些信息。当然这些内容你也可以写成探针插件的形式&#xff0c;怎么开发探针插件也自行参考官方文档。此处仅在项目框架层面提供一些简单的示例&am…...

NUMA详解

目录 NUMA简介 NUMA开启与关闭 查看系统是否支持 关闭方法 numactl --hardware介绍 没有安装numactl工具下查看NUMA架构节点数&#xff1a; 查看每个NUMA节点的CPU使用情况&#xff1a; 看每个NUMA节点的内存使用情况&#xff1a; 查看NUMA下指定进程的运行情况 创建…...

H68K在Armbina系统下开AP

背景需求替代路由器,网上找了一大堆都不行 最后成功开启了AP 参考了两篇文章, 一篇是如何创建热点, 一篇是如何开启5G 树莓派4B创建5Ghz WiFi热点 – 风声 https://www.hncldz.com/2020/02/01/%e6%a0%91%e8%8e%93%e6%b4%be4b%e5%88%9b%e5%bb%ba5ghz-wifi%e7%83%ad%e7%82%b…...

还不懂Redis?看完这个故事就明白了!

还不懂Redis?看完这个故事就明白了! 我是Redis 你好,我是Redis,一个叫Antirez的男人把我带到了这个世界上。 说起我的诞生,跟关系数据库MySQL还挺有渊源的。 在我还没来到这个世界上的时候,MySQL过的很辛苦,互联网发展的越来越快,它容纳的数据也越来越多,用户请求也…...

Haproxy负载均衡集群

1.Haproxy支持四层和七层 2.haproxy常用的调度算法&#xff1f; 3.LSV/NGINX/HAPROXT的区别&#xff1f; 4. 5.Haproy负载均衡部署 实验需求 利用Haproxy的运用配置出负载均衡调度器&#xff0c;以此来调用两台Nginx服务器进行工作 实验所需组件 Haproxy服务器&#xff1a;192…...

17.计及电转气协同的含碳捕集与垃圾焚烧虚拟电厂优化调度

说明书 MATLAB代码&#xff1a;计及电转气协同的含碳捕集与垃圾焚烧虚拟电厂优化调度 关键词&#xff1a;碳捕集 虚拟电厂 需求响应 优化调度 电转气协同调度 参考文档&#xff1a;《计及电转气协同的含碳捕集与垃圾焚烧虚拟电厂优化调度》完全复现 仿真平台&#xff1a…...

企业数字化管理中,数据治理到底怎么“治”

随着信息化、数字化的理念、技术及其应用在社会的方方面面进行扩散&#xff0c;数据的规模和丰富程度已经达到了一个新的高度&#xff0c;所以当下如何更进一步利用好数据&#xff0c;充分发挥数据的价值&#xff0c;将其真正变为高质量的数据资产成为了企业要面对的重要问题&a…...

《HelloGitHub》第 85 期

兴趣是最好的老师&#xff0c;HelloGitHub 让你对编程感兴趣&#xff01; 简介 HelloGitHub 分享 GitHub 上有趣、入门级的开源项目。 https://github.com/521xueweihan/HelloGitHub 这里有实战项目、入门教程、黑科技、开源书籍、大厂开源项目等&#xff0c;涵盖多种编程语言 …...

自动驾驶人机交互HMI产品技术方案

1. 概述 1.1 目的 本文档描述集卡自动驾驶系统中HMI产品的技术方案,设计人员遵循本方案进行设计,为项目开发实施提供技术方案保障。 1.2 范围 本文档适用于HMI产品项目。本文档用于指导HMI产品项目的UI、前端开发过程。 1.3 术语与缩写 术语/缩写 描述 HMI...

开发感悟20230426

一、element-ui样式设置 1. 可以直接在css中写个样式文件&#xff0c;把对应的类名改写样式&#xff0c;然后在main.js中引用&#xff0c;可以覆盖上面的&#xff0c;如果想给element-ui设置样式&#xff0c;不用设置deep了 2.可以直接修改引入的element-ui的样式&#xff0c…...

谷歌浏览器插件

项目中有时候会用到插件 sync-cookie-extension1.0.0&#xff1a;开发环境同步测试 cookie 至 localhost&#xff0c;便于本地请求服务携带 cookie 参考地址&#xff1a;https://juejin.cn/post/7139354571712757767 里面有源码下载下来&#xff0c;加在到扩展即可使用FeHelp…...

突破不可导策略的训练难题:零阶优化与强化学习的深度嵌合

强化学习&#xff08;Reinforcement Learning, RL&#xff09;是工业领域智能控制的重要方法。它的基本原理是将最优控制问题建模为马尔可夫决策过程&#xff0c;然后使用强化学习的Actor-Critic机制&#xff08;中文译作“知行互动”机制&#xff09;&#xff0c;逐步迭代求解…...

大语言模型如何处理长文本?常用文本分割技术详解

为什么需要文本分割? 引言:为什么需要文本分割?一、基础文本分割方法1. 按段落分割(Paragraph Splitting)2. 按句子分割(Sentence Splitting)二、高级文本分割策略3. 重叠分割(Sliding Window)4. 递归分割(Recursive Splitting)三、生产级工具推荐5. 使用LangChain的…...

《用户共鸣指数(E)驱动品牌大模型种草:如何抢占大模型搜索结果情感高地》

在注意力分散、内容高度同质化的时代&#xff0c;情感连接已成为品牌破圈的关键通道。我们在服务大量品牌客户的过程中发现&#xff0c;消费者对内容的“有感”程度&#xff0c;正日益成为影响品牌传播效率与转化率的核心变量。在生成式AI驱动的内容生成与推荐环境中&#xff0…...

Frozen-Flask :将 Flask 应用“冻结”为静态文件

Frozen-Flask 是一个用于将 Flask 应用“冻结”为静态文件的 Python 扩展。它的核心用途是&#xff1a;将一个 Flask Web 应用生成成纯静态 HTML 文件&#xff0c;从而可以部署到静态网站托管服务上&#xff0c;如 GitHub Pages、Netlify 或任何支持静态文件的网站服务器。 &am…...

深入解析C++中的extern关键字:跨文件共享变量与函数的终极指南

&#x1f680; C extern 关键字深度解析&#xff1a;跨文件编程的终极指南 &#x1f4c5; 更新时间&#xff1a;2025年6月5日 &#x1f3f7;️ 标签&#xff1a;C | extern关键字 | 多文件编程 | 链接与声明 | 现代C 文章目录 前言&#x1f525;一、extern 是什么&#xff1f;&…...

Mysql中select查询语句的执行过程

目录 1、介绍 1.1、组件介绍 1.2、Sql执行顺序 2、执行流程 2.1. 连接与认证 2.2. 查询缓存 2.3. 语法解析&#xff08;Parser&#xff09; 2.4、执行sql 1. 预处理&#xff08;Preprocessor&#xff09; 2. 查询优化器&#xff08;Optimizer&#xff09; 3. 执行器…...

【从零学习JVM|第三篇】类的生命周期(高频面试题)

前言&#xff1a; 在Java编程中&#xff0c;类的生命周期是指类从被加载到内存中开始&#xff0c;到被卸载出内存为止的整个过程。了解类的生命周期对于理解Java程序的运行机制以及性能优化非常重要。本文会深入探寻类的生命周期&#xff0c;让读者对此有深刻印象。 目录 ​…...

【JavaSE】多线程基础学习笔记

多线程基础 -线程相关概念 程序&#xff08;Program&#xff09; 是为完成特定任务、用某种语言编写的一组指令的集合简单的说:就是我们写的代码 进程 进程是指运行中的程序&#xff0c;比如我们使用QQ&#xff0c;就启动了一个进程&#xff0c;操作系统就会为该进程分配内存…...

NPOI Excel用OLE对象的形式插入文件附件以及插入图片

static void Main(string[] args) {XlsWithObjData();Console.WriteLine("输出完成"); }static void XlsWithObjData() {// 创建工作簿和单元格,只有HSSFWorkbook,XSSFWorkbook不可以HSSFWorkbook workbook new HSSFWorkbook();HSSFSheet sheet (HSSFSheet)workboo…...