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

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

观察者模式

1、观察者模式介绍


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


1.1 观察者模式结构图

  1. Subject类,可翻译为主题或抽象通知者,一般用一个抽象类或者一个接口实现。它把所有对观察者对象的引用保存在一个聚集里


,每个主题都可以有任何数量的观察者。抽象主题提供一个接口,可以增加和删除观察者对象。

Observer类,抽象观察者,为所有的具体观察者定义一个接口,在得到主题的通知时更新自己。这个接口叫作更新接口。抽象观察者一般用一个抽象类或者一个接口实现。更新接口通常包含一个update()更新方法。

ConcreteSubject类,叫作具体主题或具体通知者,将有关状态存入具体观察者对象;在具体主题的内部状态改变时,给所有登记过的观察者发出通知。具体主题角色通常用一个具体子类实现。

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、观察者模式总结

观察者模式的优点:

解耦性:观察者模式可以将观察者和主题对象解耦。主题对象只知道观察者的接口,而不需要知道具体观察者的实现。这使得主题对象和观察者可以独立地进行修改和扩展,互不影响。

可扩展性:通过添加新的观察者,可以方便地扩展系统的功能。新的观察者可以在不修改现有代码的情况下加入到系统中。

面向对象设计的灵活性:观察者模式符合面向对象设计的原则,可以实现低耦合、高内聚的设计。

观察者模式的缺点:


如果观察者过多或者观察者的处理逻辑复杂,会导致通知过程的效率降低。

观察者模式可能会导致循环引用的问题。当观察者和主题对象相互引用时,需要注意处理引用关系,避免出现内存泄漏的情况。

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


当一个对象的改变需要同时通知多个其他对象时,可以使用观察者模式。例如,当一个数据模型改变时,需要通知多个视图进行更新。

当一个抽象模型有两个方面,其中一个方面依赖于另一个方面,而且这两个方面都需要独立地改变和扩展时,可以使用观察者模式。例如,一个股票市场的模型中,股票价格的变化会影响到显示股票价格的数字和显示股票价格的图形等多个视图。

当一个对象的改变需要触发其他多个对象的更新操作时,可以使用观察者模式。例如,一个订单对象状态的改变需要触发库存管理、支付系统、通知系统等多个模块的更新。


目录
相关文章
|
19天前
|
设计模式 监控 Java
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
|
1月前
|
设计模式 传感器
【设计模式】观察者模式(定义 | 特点 | Demo入门讲解)
【设计模式】观察者模式(定义 | 特点 | Demo入门讲解)
40 0
|
15天前
|
设计模式 监控 Java
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
25 3
|
26天前
|
设计模式 监控 Java
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
34 9
|
25天前
|
设计模式 监控 Java
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
29 2
|
1月前
|
设计模式 监控 UED
设计模式之观察者模式
【10月更文挑战第12天】 观察者模式是一种行为型设计模式,定义了一对多的依赖关系,当一个对象状态改变时,所有依赖它的对象都会自动更新。主要由主题(被观察者)和观察者组成,实现对象间的松耦合,广泛应用于用户界面、事件驱动系统和数据监控等领域。
|
1月前
|
设计模式 监控 Java
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
本教程详细讲解Kotlin语法,适合深入学习。对于快速掌握Kotlin,推荐“简洁”系列教程。本文特别介绍了观察者模式,包括使用Java API和Kotlin委托属性(如Delegates.observable)实现的方法,旨在帮助开发者更高效地实现和优化观察者模式的应用。
34 3
|
2月前
|
设计模式 Java 关系型数据库
设计模式——观察者模式
观察者模式介绍、观察者模式优化天气预报案例、JDK 的Observable类和Observer类
设计模式——观察者模式
|
1月前
|
设计模式 监控 Java
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
Kotlin教程笔记(52) - 改良设计模式 - 观察者模式
32 0
|
3月前
|
设计模式 存储 前端开发
【十四】设计模式~~~行为型模式~~~观察者模式(Java)
文章详细介绍了观察者模式(Observer Pattern),这是一种对象行为型模式,用于建立对象之间的一对多依赖关系。当一个对象状态发生改变时,所有依赖于它的对象都会得到通知并自动更新。文中通过交通信号灯与汽车的案例以及多人联机对战游戏的设计方案,阐述了观察者模式的动机和应用场景。接着,文章介绍了观察者模式的结构、角色、优点、缺点以及适用情况,并通过代码示例展示了如何在Java中实现观察者模式。此外,还探讨了观察者模式在MVC架构中的应用以及Java中对观察者模式的支持。
【十四】设计模式~~~行为型模式~~~观察者模式(Java)

热门文章

最新文章

  • 1
    C++一分钟之-设计模式:工厂模式与抽象工厂
    43
  • 2
    《手把手教你》系列基础篇(九十四)-java+ selenium自动化测试-框架设计基础-POM设计模式实现-下篇(详解教程)
    48
  • 3
    C++一分钟之-C++中的设计模式:单例模式
    58
  • 4
    《手把手教你》系列基础篇(九十三)-java+ selenium自动化测试-框架设计基础-POM设计模式实现-上篇(详解教程)
    38
  • 5
    《手把手教你》系列基础篇(九十二)-java+ selenium自动化测试-框架设计基础-POM设计模式简介(详解教程)
    63
  • 6
    Java面试题:结合设计模式与并发工具包实现高效缓存;多线程与内存管理优化实践;并发框架与设计模式在复杂系统中的应用
    58
  • 7
    Java面试题:设计模式在并发编程中的创新应用,Java内存管理与多线程工具类的综合应用,Java并发工具包与并发框架的创新应用
    42
  • 8
    Java面试题:如何使用设计模式优化多线程环境下的资源管理?Java内存模型与并发工具类的协同工作,描述ForkJoinPool的工作机制,并解释其在并行计算中的优势。如何根据任务特性调整线程池参数
    50
  • 9
    Java面试题:请列举三种常用的设计模式,并分别给出在Java中的应用场景?请分析Java内存管理中的主要问题,并提出相应的优化策略?请简述Java多线程编程中的常见问题,并给出解决方案
    110
  • 10
    Java面试题:设计模式如单例模式、工厂模式、观察者模式等在多线程环境下线程安全问题,Java内存模型定义了线程如何与内存交互,包括原子性、可见性、有序性,并发框架提供了更高层次的并发任务处理能力
    78
  • 下一篇
    无影云桌面