行为型设计模式03-观察者模式

news2024/11/15 23:55:29

🧑‍💻作者:猫十二懿

🏡账号:CSDN 、个人博客 、Github

🎊公众号:猫十二懿

观察者模式

1、观察者模式介绍

观察者模式是一种行为型设计模式,也被称为发布-订阅模式,它定义了一种一对多的依赖关系,当一个对象状态发生改变时,其所有的依赖对象都会得到通知并自动更新。也就是说让多个观察者对象同时监听某一个主题对象。这个主题对象在状态发生变化时,会通知所有观察者对象,使它们能够自动更新自己。

1.1 观察者模式结构图

在这里插入图片描述

  1. Subject类,可翻译为主题或抽象通知者,一般用一个抽象类或者一个接口实现。它把所有对观察者对象的引用保存在一个聚集里,每个主题都可以有任何数量的观察者。抽象主题提供一个接口,可以增加和删除观察者对象。
  2. Observer类,抽象观察者,为所有的具体观察者定义一个接口,在得到主题的通知时更新自己。这个接口叫作更新接口。抽象观察者一般用一个抽象类或者一个接口实现。更新接口通常包含一个update()更新方法。
  3. ConcreteSubject类,叫作具体主题或具体通知者,将有关状态存入具体观察者对象;在具体主题的内部状态改变时,给所有登记过的观察者发出通知。具体主题角色通常用一个具体子类实现。
  4. ConcreteObserver类,具体观察者,实现抽象观察者角色所要求的更新接口,以便使本身的状态与主题的状态相协调。具体观察者角色可以保存一个指向具体主题对象的引用。具体观察者角色通常用一个具体子类实现。

抽象例子如下:

Subject类:

/**
 * @author Shier
 * CreateTime 2023/4/26 22:48
 * 抽象通知者类
 */
public abstract class Subject {
    protected String subjectState;

    public String getSubjectState() {
        return subjectState;
    }

    public void setSubjectState(String subjectState) {
        this.subjectState = subjectState;
    }

    private ArrayList<Observer> list = new ArrayList<Observer>();

    /**
     * 增加观察者
     *
     * @param observer
     */
    public void attach(Observer observer) {
        list.add(observer);
    }

    /**
     * 减少观察者
     *
     * @param observer
     */
    public void detach(Observer observer) {
        list.remove(observer);
    }

    /**
     * 通知观察者
     */
    public void notifyObserver() {
        for (Observer observer : list) {
            observer.update();
        }
    }
}

Observer类:

/**
 * @author Shier
 * CreateTime 2023/4/26 22:49
 * 抽象观察者
 */
public abstract class Observer {
    public abstract void update();
}

ConcreteObserver类:

/**
 * @author Shier
 * CreateTime 2023/4/26 22:55
 */
public class ConcreteObserver extends Observer{
    private String name;
    private Subject subject;

    public ConcreteObserver(String name, Subject subject) {
        this.name = name;
        this.subject = subject;
    }

    @Override
    public void update() {
        System.out.println("观察者"+this.name+"的最新状态是"+this.subject.subjectState);
    }
}

ConcreteSubject类:

/**
 * @author Shier
 * CreateTime 2023/4/26 22:54
 * 具体通知者
 */
public class ConcreteSubject extends Subject {
    // 具体的通知方法
}

测试类:

/**
 * @author Shier
 * CreateTime 2023/4/26 22:57
 */
public class ObserverTest {
    public static void main(String[] args) {
        Subject subject = new ConcreteSubject();
        subject.attach(new ConcreteObserver("shier1", subject));
        subject.attach(new ConcreteObserver("shier2", subject));
        subject.attach(new ConcreteObserver("shier3", subject));
        subject.attach(new ConcreteObserver("shier4", subject));
        subject.setSubjectState("睡觉了");
        // 通知其他观察者
        subject.notifyObserver();
    }
}

测试结果:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-yZOg9QDW-1685960128875)(…/…/…/…/AppData/Roaming/Typora/typora-user-images/image-20230426230307411.png)]

tSubjectState(“睡觉了”);
// 通知其他观察者
subject.notifyObserver();
}
}




## 2、观察者模式具体例子

> 假设我们有一个电商平台,用户可以在该平台上购买商品。在这个场景下,我们可以使用观察者模式来实现购物车功能

首先,我们需要定义两个角色:主题(Subject)和观察者(Observer)。主题就是购物车,而观察者就是用户。

```java
// 主题接口
interface Subject {
    void registerObserver(Observer observer);
    void removeObserver(Observer observer);
    void notifyObservers();
}

// 观察者接口
interface Observer {
    void update(String itemName);
}
// 购物车类作为主题实现Subject接口
class ShoppingCart implements Subject {
    private List<Observer> observers = new ArrayList<>();
    private List<String> items = new ArrayList<>();

    public void addItem(String itemName) {
        items.add(itemName);
        notifyObservers();
    }

    public void removeItem(String itemName) {
        items.remove(itemName);
        notifyObservers();
    }

    @Override
    public void registerObserver(Observer observer) {
        observers.add(observer);
    }

    @Override
    public void removeObserver(Observer observer) {
        observers.remove(observer);
    }

    @Override
    public void notifyObservers() {
        for (Observer observer : observers) {
            observer.update(items.get(items.size() - 1));
        }
    }
}
// 用户类作为观察者实现Observer接口
class User implements Observer {
    private String name;

    public User(String name) {
        this.name = name;
    }

    @Override
    public void update(String itemName) {
        System.out.println(name + " 收到通知:购物车中添加了商品 " + itemName);
    }
}
// 测试代码
public class ObserverPatternExample {
    public static void main(String[] args) {
        ShoppingCart cart = new ShoppingCart();

        User user1 = new User("Alice");
        User user2 = new User("Bob");

        cart.registerObserver(user1);
        cart.registerObserver(user2);

        cart.addItem("手机");
        cart.addItem("电视");

        cart.removeItem("手机");
    }
}

在上面的示例中,购物车(ShoppingCart)作为主题实现了Subject接口。它维护了一个观察者列表,并提供了注册、移除和通知观察者的方法。

用户(User)作为观察者实现了Observer接口。每当购物车的状态发生改变时,购物车会通知所有注册的观察者,观察者接收到通知后可以执行相应的操作。

在测试代码中,我们创建了一个购物车对象(cart),以及两个用户对象(user1和user2)。首先,我们将两个用户注册为购物车的观察者。然后,我们通过调用addItem方法向购物车中添加商品,购物车会通知所有观察者。观察者收到通知后会打印相应的信息。最后,我们通过调用removeItem方法从购物车中移除商品,购物车再次通知观察者。

3、观察者模式总结

观察者模式的优点:

  1. 解耦性:观察者模式可以将观察者和主题对象解耦。主题对象只知道观察者的接口,而不需要知道具体观察者的实现。这使得主题对象和观察者可以独立地进行修改和扩展,互不影响。
  2. 可扩展性:通过添加新的观察者,可以方便地扩展系统的功能。新的观察者可以在不修改现有代码的情况下加入到系统中。
  3. 面向对象设计的灵活性:观察者模式符合面向对象设计的原则,可以实现低耦合、高内聚的设计。

观察者模式的缺点:

  1. 如果观察者过多或者观察者的处理逻辑复杂,会导致通知过程的效率降低。
  2. 观察者模式可能会导致循环引用的问题。当观察者和主题对象相互引用时,需要注意处理引用关系,避免出现内存泄漏的情况。

观察者模式适用于以下场景:

  1. 当一个对象的改变需要同时通知多个其他对象时,可以使用观察者模式。例如,当一个数据模型改变时,需要通知多个视图进行更新。
  2. 当一个抽象模型有两个方面,其中一个方面依赖于另一个方面,而且这两个方面都需要独立地改变和扩展时,可以使用观察者模式。例如,一个股票市场的模型中,股票价格的变化会影响到显示股票价格的数字和显示股票价格的图形等多个视图。
  3. 当一个对象的改变需要触发其他多个对象的更新操作时,可以使用观察者模式。例如,一个订单对象状态的改变需要触发库存管理、支付系统、通知系统等多个模块的更新。

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

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

相关文章

Hive学习---4、函数(单行函数、高级聚合函数、炸裂函数、窗口函数)

1、函数 1.1 函数简介 Hive会将常用的逻辑封装成函数给用户进行使用&#xff0c;类似java中的函数。 好处&#xff1a;避免用户反复写逻辑&#xff0c;可以直接拿来使用 重点&#xff1a;用户需要知道函数叫什么&#xff0c;能做什么 Hive提供了大量的内置函数&#xff0c;按…

《相信》读后感

近日阅读了蔡磊的《相信》一书&#xff0c;蔡磊先生曾是京东集团副总裁&#xff0c;中国电子发票的推动者。上天给了他优越的智商条件&#xff0c;从上学到工作&#xff0c;前半生几乎顺风顺水、获誉无数&#xff0c;却在初为人父、本该享受家庭幸福的时候&#xff0c;接到突患…

Python字典及用法详解

Python中的字典&#xff08;Dictionary&#xff09;是一种无序、可变的数据类型&#xff0c;用于存储键&#xff08;Key&#xff09;和值&#xff08;Value&#xff09;之间的映射关系。字典是一种高效的数据结构&#xff0c;可以用于快速查找和检索数据。 1.创建字典 可以使…

MobileViT详解:轻型,通用,移动友好的视觉变压器

MobileViT详解&#xff1a;轻型&#xff0c;通用&#xff0c;移动友好的视觉变压器 0. 引言1. 网络结构2. 模型详解2.1 MobileViT Block2.1.1 Local representations2.1.2 Transformers as Convolutions (global representations)2.1.3 Fusion 2.2 MV2 3. 简化版理解4. 总结 0.…

Ubuntu系统搭建FTP服务器

Ubuntu 系统版本&#xff1a;Ubuntu 22.04.2 LTS 安装 vsftpd 软件包 sudo apt-get update sudo apt-get install vsftpd查看版本&#xff0c;验证是否安装成功&#xff1a;vsftpd -v 配置文件 以下是我翻译后的默认配置文件&#xff08;地址 /etc/vsftpd.conf&#xff09;&a…

[NOI2009] 描边

题目描述 小 Z 是一位杰出的数学家。聪明的他特别喜欢研究一些数学小问题。 有一天&#xff0c;他在一张纸上选择了 n 个点&#xff0c;并用铅笔将它们两两连接起来&#xff0c;构成 (&#xfffd;−1)22n(n−1)​ 条线段。由于铅笔很细&#xff0c;可以认为这些线段的宽度为…

ROS:参数的使用与编程方法

目录 一、参数模型二、 创建功能包三、参数命令行的使用(rosparam)四、使用程序来使用参数&#xff08;C&#xff09;4.1创建代码4.2编译4.3运行 一、参数模型 在ROS Master中&#xff0c;存在一个参数服务器&#xff08;Parameter Server&#xff09;&#xff0c;它是一个全局…

Python高光谱遥感数据处理与机器学习实践技术丨Matlab高光谱遥感数据处理与混合像元分解

目录 Python高光谱遥感数据处理与机器学习实践技术 第一章 高光谱基础 第二章 高光谱开发基础&#xff08;Python&#xff09; 第三章 高光谱机器学习技术&#xff08;python&#xff09; 第四章 典型案例操作实践 Matlab 高光谱遥感数据处理与混合像元分解 第一章 理论…

java SSM 互助旅游管理系统myeclipse开发mysql数据库springMVC模式java编程计算机网页设计

一、源码特点 java SSM 互助旅游管理系统是一套完善的web设计系统&#xff08;系统采用SSM框架进行设计开发&#xff0c;springspringMVCmybatis&#xff09;&#xff0c;对理解JSP java编程开发语言有帮助&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统主要采…

预制菜进击万亿市场,谁能更快上桌“吃菜”?

文 | 螳螂观察 作者 | 图霖 消费行业很少有可持续的风口&#xff0c;这两年的预制菜算其中一个。 艾媒咨询发布的行业预测显示&#xff0c;2026年我国预制菜市场规模有望达到10720亿元。 过去这一年&#xff0c;武汉、大同等地已相继召开了预制菜相关的产业峰会。峰会规模有…

gismo-3维IGA

文章目录 前言一、简单示例二、gismo-3维IGA3维程序中的几何模型 三、xml文件的理解1、xml文件示例2、gismo中二维示例文件-一个曲面&#xff08;简单&#xff09; 四、三维程序中xml文件的理解三维几何模型边界信息 五、三维程序运行细化四次细化5次 总结 #pic_center 前言 只…

C#读写FDX-B ISO11784/85协议动物标签源码

一个FDX-B ISO11784/85协议动物标签内包括了以下信息&#xff1a; 11位的前导码&#xff1b;38位的SN序号&#xff1b;10位国家代码&#xff1b;1位data block标识&#xff1b;14位保留位&#xff1b;1位Animal动物标识&#xff1b;以上64位数据的crc16ccitt校验码&#xff0c…

短视频矩阵源码系统打包.源码

Masayl是一款基于区块链技术的去中心化应用程序开发平台&#xff0c;可帮助开发者快速、便捷地创建去中心化应用程序。Masayl拥有丰富的API和SDK&#xff0c;为开发者们提供了支持。此外&#xff0c;Masayl还采用了高效的智能合约技术&#xff0c;确保应用程序的稳定、安全和高…

项目集管理—项目集治理

一、概述 项目集治理是实现和执行项目集决策&#xff0c;为支持项目集而制定实践&#xff0c;并维持项目集监督的绩效领域。 本章包括&#xff1a; 项目集治理实践项目集治理角色项目集治理设计与实施 项目集治理包括为了满足组织战略和运营目标的要求&#xff0c;对项目集实…

【虹科案例】虹科数字化仪在激光雷达大气研究中的应用

01 莱布尼茨研究所使用激光雷达进行大气研究 图 1&#xff1a;在 Khlungsborn 的 IAP 办公室测试各种激光器 大气研究使用脉冲激光束通过测量大气中 100 公里高度的多普勒频移和反向散射光来测量沿光束的温度和风速。返回的光信号非常微弱&#xff0c;会被阳光阻挡&#xff0c…

90后的心声:都别卷了,上年纪了真的卷不动.....

内卷&#xff0c;是现在热度非常高的一个词汇&#xff0c;随着热度不断攀升&#xff0c;隐隐到了“万物皆可卷”的程度。 内卷的来源 内卷最早的“出处”是几张名校学霸的图片。 大学生们刷爆朋友圈的几张“内卷”图片是这样的&#xff1a;有的人骑在自行车上看书&#xff0c…

如何处理亿级图片排重(精准排重,相似排重)

图片相似度对比 1、需求 假如有一个图片池&#xff0c;存有1亿图片。给一张目标图片&#xff0c;在图片池中做匹配。 判断一张图片是否在图片池中出现过。&#xff08;完全一样&#xff09;判断有没有相似的出现过。比如两张图相似度90&#xff0c;两张图片是在描述一件事情。 …

系统架构师之高内聚低耦合

一、概念&#xff1a; 标记耦合&#xff08;Stamp Coupling&#xff09;和数据耦合&#xff08;Data Coupling&#xff09;是软件设计中两种不同的耦合类型&#xff0c;它们之间的区别如下&#xff1a; 标记耦合&#xff1a;标记耦合是指模块之间通过参数传递标记或标识符来进…

对安装Linux的服务器进行缓存清除

安装Linux的服务器缓存过高导致服务器运行速度慢 第一步&#xff0c;查看当前服务器中Linux系统的状态。 命令&#xff1a;free -h 第二步&#xff1a;备份内存缓冲区中的数据到磁盘中。 命令&#xff1a;sync 在大多数情况下&#xff0c;不需要手动使用sync命令&#xff0…

跨境电商如何进行仓储物流管理?

跨境电商如何进行仓储物流管理&#xff1f; 01跨境电商仓储物流管理痛点在哪&#xff1f; 供应链不稳定&#xff1a;因为要涉及多个国家的生产和供应环节&#xff0c;跨境物流的过程中还需要遵守目的地国家和货物品类的规定&#xff0c;这会增加仓储和物流成本&#xff0c;并…