【设计模式-4.1】行为型——观察者模式

说明:本文介绍设计模式中行为型设计模式中的,观察者模式;

商家与顾客

观察者模式属于行为型设计模式,关注对象的行为。以商家与顾客为例,商家有商品,顾客来购买商品,如果商家商品卖完了,顾客则购买失败。如下:

(Shop,商家类,提供购买、进货方法)

/*** 商家*/
public class Shop {/*** 商品*/private String product;public Shop() {this.product = "无商品";}/*** 出货* @return*/public String getProduct() {return product;}/*** 进货* @param product*/public void setProduct(String product) {this.product = product;}
}

(Buyer,顾客类,注入商店对象,购买商店商品)

/*** 顾客*/
public class Buyer {private String name;private Shop shop;public Buyer(String name, Shop shop) {this.name = name;this.shop = shop;}/*** 买家购买商品*/public void buy() {System.out.println(name + "来购买:" + shop.getProduct());}
}

(Client,客户端,演示购买动作)

/*** 客户端*/
public class Client {public static void main(String[] args) {Shop shop = new Shop();new Buyer("张三", shop).buy();shop.setProduct("橘子");new Buyer("李四", shop).buy();}
}

(执行结果)

在这里插入图片描述

分析上面的行为,可以发现一点缺陷。当商家的商品发生变化时,无法主动的,自发的通知顾客,只有顾客来购买时,才知道商品的信息。因此,我们考虑是否可以在商家类内部维护一个“顾客列表”,当商品信息发生变化时,调用顾客列表中所有顾客对应的方法,达到主动通知客户的目的。如下:

(Shop,商家类,维护一个顾客列表,新增一个通知顾客的方法)

import java.util.ArrayList;
import java.util.List;/*** 商家*/
public class Shop {/*** 商品*/private String product;/*** 顾客列表*/private List<Buyer> buyers;public Shop() {this.product = "无商品";this.buyers = new ArrayList<>();}/*** 添加顾客* @param buyer*/public void addBuyer(Buyer buyer) {this.buyers.add(buyer);}/*** 移除顾客* @param buyer*/public void removeBuyer(Buyer buyer) {this.buyers.remove(buyer);}/*** 出货* @return*/public String getProduct() {return product;}/*** 进货* @param product*/public void setProduct(String product) {this.product = product;notifyBuyers();}/*** 通知顾客*/public void notifyBuyers() {for (Buyer buyer : buyers) {buyer.buy(product);}}
}

(Buyer,顾客抽象类,创建一个购买商品的抽象方法)

/*** 顾客*/
public abstract class Buyer {/*** 顾客姓名*/protected String name;public Buyer(String name) {this.name = name;}/*** 买家购买商品*/public abstract void buy(String product);
}

(FruitBuyer,水果买家,只购买水果类的商品)

/*** 水果买家*/
public class FruitBuyer extends Buyer{public FruitBuyer(String name) {super(name);}@Overridepublic void buy(String product) {if (product.contains("水果")) {System.out.println(name + ":来购买了水果");}}
}

(PhoneBuyer,手机买家,只购买手机类的商品)

/*** 手机买家*/
public class PhoneBuyer extends Buyer{public PhoneBuyer(String name) {super(name);}@Overridepublic void buy(String product) {if (product.contains("手机")) {System.out.println(name + ":来购买了手机");}}
}

(Client,客户端,模拟购买行为,张三只购买手机,买了一次后就不再需要了)

/*** 客户端*/
public class Client {public static void main(String[] args) {Shop shop = new Shop();PhoneBuyer phoneBuyer = new PhoneBuyer("张三");FruitBuyer fruitBuyer = new FruitBuyer("李四");shop.addBuyer(phoneBuyer);shop.addBuyer(fruitBuyer);shop.setProduct("苹果手机");shop.setProduct("各种水果");shop.removeBuyer(phoneBuyer);System.out.println("=================水果和手机降价了=================");shop.setProduct("水果和手机降价了");}
}

执行结果。当商家的商品信息有变化时,会通知到对应的顾客来购买。

在这里插入图片描述

类比现实生活,小村庄里只有一家商店。我们需要购买某样日用品,去商店购买,店长说没有,下个月才进货。很自然的,我们会和店长说,那我留下电话,进货了打电话通知我。

其中,留下电话,可以类比为将顾客注册到商店类中,而打电话则是调用顾客类对应的方法。

Spring中的体现

观察者模式,是一种订阅-发布的模式。在上面的例子中,商家是发布者,顾客是订阅者,当商家商品信息有变动时,发布信息,顾客接收。

Spring框架中,就有观察者模式的应用。如下:

(Customer,消费者类)

import org.springframework.context.ApplicationEvent;public class Customer extends ApplicationEvent {private String message;public Customer(Object source, String message) {super(source);this.message = message;}public String getMessage() {return message;}
}

(CustomListener1,消费者1号)

import org.springframework.context.ApplicationListener;
import org.springframework.stereotype.Component;@Component
public class CustomListener1 implements ApplicationListener<Customer> {@Overridepublic void onApplicationEvent(Customer customer) {System.out.println("1号订阅者收到消息: " + customer.getMessage());}
}

(CustomListener2,消费者2号)

import org.springframework.context.ApplicationListener;
import org.springframework.stereotype.Component;@Component
public class CustomListener2 implements ApplicationListener<Customer> {@Overridepublic void onApplicationEvent(Customer customer) {System.out.println("2号订阅者收到消息: " + customer.getMessage());}
}

(Publisher,发布者)

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationEventPublisher;
import org.springframework.stereotype.Component;@Component
public class Publisher {@Autowiredprivate ApplicationEventPublisher eventPublisher;public void publishCustomEvent(String message) {eventPublisher.publishEvent(new Customer(this, message));}
}

(在测试类中,发布信息)

import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;@SpringBootTest
@RunWith(SpringRunner.class)
public class PublisherTest {@Autowiredprivate Publisher publisher;@Testpublic void test1() {publisher.publishCustomEvent("Hello World!");}
}

执行结果,2个订阅者都收到了消息;

在这里插入图片描述

以上就是观察者模式的实现;

总结

本文参考《设计模式的艺术》、《秒懂设计模式》两书

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

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

相关文章

Vue3中teleport如何使用

Vue 3作为一种流行的JavaScript框架&#xff0c;一直以来都在努力提供更便捷、灵活的开发体验。本文将深入解析Teleport&#xff0c;包括其详细的使用方法、源码实现机制以及在实际项目中的应用场景。 一、Teleport是什么&#xff1f; Vue 3中的Teleport允许开发者将组件的内容…

Vue+ElementUI+C#前后端分离:监控长耗时任务的实践

想象一下&#xff0c;我们正在构建一个Web应用&#xff0c;需要实现一个数据报告的导出功能。这听起来很简单&#xff0c;不是吗&#xff1f;但是&#xff0c;随着深入开发&#xff0c;我们意识到导出过程比预期的要复杂和耗时得多。由于报告的数据量巨大&#xff0c;后端需要花…

PostgreSQL有意思的现象:支持不带列的表

1、前言 以前从没有试过建一张表&#xff0c;不带任何列。在PG中却支持这种语法。这是个什么鬼? 最近&#xff0c;把PG源码扒了下&#xff0c;简单浏览了下最近的一些merge。其中有一个fix&#xff1a; eeb0ebad79 ("Fix the initial sync tables with no columns.&qu…

〖大前端 - 基础入门三大核心之JS篇㊺〗- 定时器和延时器

说明&#xff1a;该文属于 大前端全栈架构白宝书专栏&#xff0c;目前阶段免费&#xff0c;如需要项目实战或者是体系化资源&#xff0c;文末名片加V&#xff01;作者&#xff1a;不渴望力量的哈士奇(哈哥)&#xff0c;十余年工作经验, 从事过全栈研发、产品经理等工作&#xf…

LeetCode 每日一题 2023/11/27-2023/12/3

记录了初步解题思路 以及本地实现代码&#xff1b;并不一定为最优 也希望大家能一起探讨 一起进步 目录 11/27 907. 子数组的最小值之和11/28 1670. 设计前中后队列11/29 2336. 无限集中的最小数字11/30 1657. 确定两个字符串是否接近12/1 2661. 找出叠涂元素12/2 1094. 拼车12…

本地缓存和分布式缓存

一、引言 在当今的大数据时代&#xff0c;数据缓存已成为提升应用性能和效率的重要策略。缓存能够降低数据访问延迟&#xff0c;提高系统响应速度&#xff0c;从而改善用户体验。根据存储位置和应用场景的不同&#xff0c;缓存技术分为本地缓存和分布式缓存两种。本文将详细介绍…

【Rust日报】2023-12-01 KCL v0.7 版本发布

RFC&#xff1a;在选择依赖项时使 Cargo 遵循最低支持的 Rust 版本 (MSRV) 概括内容是&#xff0c;为需要使用旧版本 Rust 的开发者提供了一条快乐之路&#xff0c;具体方法是&#xff1a; 在 Cargo 解析依赖关系时&#xff0c;优先选择与 MSRV&#xff08;最低支持的 Rust 版本…

如何使用PHPUnit编写一个PHP单元测试-简单的代码示例

在软件开发过程中&#xff0c;单元测试是一种重要的测试方法&#xff0c;可以确保代码的质量和可靠性。在PHP开发中&#xff0c;也可以通过编写单元测试来验证代码的正确性。下面将介绍一些编写PHP单元测试的基本步骤和常用工具。 首先&#xff0c;你需要一个PHP单元测试框架&…

3D场景建模工具

在线工具推荐&#xff1a; 三维数字孪生场景工具 - GLTF/GLB在线编辑器 - Three.js AI自动纹理化开发 - YOLO 虚幻合成数据生成器 - 3D模型在线转换 - 3D模型预览图生成服务 1. 什么是3D场景建模&#xff1f; 3D场景建模是一种通过计算机图形学技术&#xff0c;将现实世…

python查询、处理、批量存入数据

1、安装数据库连接器 首先需要安装一个数据库连接器&#xff0c;比如pymysql、pyodbc等&#xff0c;用于连接MySQL、SQL Server等不同的数据库。 安装命令如下 pip install PyMySQL2. 连接数据库 连接数据库需要先指定数据库的主机名、端口号、用户名和密码等信息。这些信息…

Vue组件分装之$attrs、$listener传递属性及事件

使用v-bind"$attrs"来将父组件的属性传递给自定义按钮 使用v-on"$listeners"将父组件的事件监听器传递给自定义按钮。 使用$slots获取父组件所有插槽以及作用域插槽对应的参数#[name]"scopeData" 这样&#xff0c;自定义按钮就能够直接响应父…

Java系列 之除字符串中的空格(trim())

我 | 在这里 &#x1f575;️ 读书 | 长沙 ⭐软件工程 ⭐ 本科 &#x1f3e0; 工作 | 广州 ⭐ Java 全栈开发&#xff08;软件工程师&#xff09; &#x1f383; 爱好 | 研究技术、旅游、阅读、运动、喜欢流行歌曲 ✈️已经旅游的地点 | 新疆-乌鲁木齐、新疆-吐鲁番、广东-广州…

【Matlab】如何快速入门一项新技能-以Matlab/Simulink入门为例

目录 1. 引言 2. 背景 3. 快速学习并完成开发 3.1 了解需求&#xff0c;知道要干什么 3.2 了解Matlab/Simulink基本功能 第一步&#xff0c;查看Matlab的中文网站中文网站https://www.ilovematlab.cn/resources/对Matlab/Simulink有了一个初步认识。 3.3 实现一个最简单…

uniapp 在app端 使用webview进行数据交互。

使用案例 1.app端(需要使用nvue) <template> <view class"webview-box"> <button style"z-index: 999;" click"handlePostMessage(app向url传值)">点击向url传值</button><web-view ref"webview" clas…

PyQt6 QDialogButtonBox组合按钮控件

锋哥原创的PyQt6视频教程&#xff1a; 2024版 PyQt6 Python桌面开发 视频教程(无废话版) 玩命更新中~_哔哩哔哩_bilibili2024版 PyQt6 Python桌面开发 视频教程(无废话版) 玩命更新中~共计34条视频&#xff0c;包括&#xff1a;2024版 PyQt6 Python桌面开发 视频教程(无废话版…

【CTA认证】Android CTA资料及信息安全要求

认证资料需求 1. 说明书&#xff1b;需有应用场景、使用人群说明 2. 产品铭牌&#xff08;需有IMEI号&#xff0c;产品名称需是&#xff1a;TD-LTE无线数据终端&#xff09; 3. 产品整体尺寸长宽高 4. 原理框图&#xff1b; 5. 主板正反面照片&#xff08;需拆除屏蔽罩&a…

快速创建桌面端(electron-egg)

介绍 | electron-egg electron-egg: 一个入门简单、跨平台、企业级桌面软件开发框架。 electron-egg是一个基于Electron和Egg.js的框架&#xff0c;可以用于快速构建跨平台的桌面应用程序。 1.兼容平台&#xff1a;electron-egg可以在Windows、MacOS和Linux等多个平台上运行…

【开源威胁情报挖掘1】引言 + 开源威胁情报挖掘框架 + 开源威胁情报采集与识别提取

基于开源信息平台的威胁情报挖掘综述 写在最前面摘要1 引言近年来的一些新型网络安全威胁类型挖掘网络威胁的情报信息威胁情报分类&#xff1a;内、外部威胁情报国内外开源威胁情报挖掘分析工作主要贡献研究范围和方法 2 开源威胁情报挖掘框架1. 开源威胁情报采集与识别2. 开源…

软件生命周期四个阶段SDLC

软件产品生命周期&#xff1a;指软件产品研发全部过程、活动和任务的结构框架。 产品的生命周期一般包括四个阶段&#xff1a;引入期、成长期、成熟期和衰退期&#xff0c;在不同的阶段中&#xff0c;市场对产品的反应不同&#xff0c;其销售特点不同&#xff0c;因而产品管理的…

mysql数据库的配置文件在哪里

可以搜索my.ini、或者my.cnf&#xff0c;看看在哪个地方。 例如&#xff0c;我在windows系统装的mysql 8.2版本&#xff0c;my.ini文件不在安装目录下&#xff0c;而在另外一个目录下。 我的安装目录是F:\Program Files\MySQL\MySQL Server 8.2&#xff0c;但my.ini文件在C:\Pr…