设计模式——观察者模式

观察者模式(Observer Pattern)是一种行为型设计模式,它定义了对象之间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都会得到通知并被自动更新。

原理

  • 主题(Subject/发布者): 提供注册和移除观察者的接口,并且在自身状态变化时负责通知所有已注册的观察者。
  • 具体主题(Concrete Subject): 具体实现主题类,通常包含具体的业务逻辑及状态,并持有观察者列表。
  • 观察者(Observer): 定义了一个更新接口,当接收到主题的通知时调用此接口进行相应的操作。
  • 具体观察者(Concrete Observer): 实现观察者接口,其中包含了根据主题状态变化需要做出反应的具体方法。

Java代码示例

java 复制代码
// 观察者接口
import java.util.ArrayList;
import java.util.List;

public interface Observable {
    void addObserver(Observer observer);
    void removeObserver(Observer observer);
    void notifyObservers(String message);
}

// 具体主题类
public class NewsAgency implements Observable {
    private List<Observer> observers = new ArrayList<>();
    private String news;

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

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

    @Override
    public void notifyObservers(String message) {
        for (Observer observer : observers) {
            observer.update(message);
        }
    }

    public void setNews(String news) {
        this.news = news;
        // 发布新消息时,通知所有观察者
        notifyObservers(news);
    }
}

// 观察者接口
public interface Observer {
    void update(String message);
}

// 具体观察者类
public class Newspaper implements Observer {
    private String name;

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

    @Override
    public void update(String message) {
        System.out.println(name + " received news: " + message);
        // 在这里执行实际的消息处理逻辑
    }
}

// 客户端使用示例
public class Client {
    public static void main(String[] args) {
        NewsAgency agency = new NewsAgency();
        Observer newspaper1 = new Newspaper("The Daily");
        Observer newspaper2 = new Newspaper("The Times");

        agency.addObserver(newspaper1);
        agency.addObserver(newspaper2);

        agency.setNews("Breaking news!");  // 当发布新消息时,会自动通知所有报纸
    }
}

想象你是一个新闻机构,有很多家报纸订阅你的新闻。每次有新的新闻时,你需要逐一打电话给每一家报纸告诉他们最新消息。在程序世界里,新闻机构就是"主题",而各家报纸是"观察者"。当新闻机构有新消息时,只需调用notifyObservers()方法,就像打一圈电话一样,所有订阅的报纸(观察者)就会自动得到通知并更新自己的内容。

应用

  • 事件驱动编程:如GUI应用中按钮点击事件触发的响应处理。
  • 订阅系统:用户订阅新闻、博客或邮件提醒,当有新内容发布时,系统自动发送通知给所有订阅者。
  • 数据同步:多个模块需要共享同一份数据源,当数据源发生变化时,各模块能够及时得到更新。

适用性

  • 当一个对象的改变需要同时改变其他对象,而且不知道具体有多少对象需要改变时。
  • 当一个对象必须通知其他对象,而并不希望指定这些对象或这些对象只知道有改变发生,而不关心如何得知这一改变时。
相关推荐
_院长大人_7 分钟前
Spring Boot 客户端设计示例:自动刷新 Token 并重试接口调用(Springboot Starter 封装)
java·spring boot·后端
卷到起飞的数分13 分钟前
19.Spring Boot原理1
java·spring boot·后端
消失的旧时光-194315 分钟前
彻底理解 synchronized:实例锁、类锁与自定义锁的原理和最佳实践
java·开发语言
开源之眼26 分钟前
github star 较多的Java双亲委派机制【类加载的核心内容加星】
java
编程火箭车28 分钟前
【Java SE 基础学习打卡】19 运算符(中)
java·java入门·运算符·编程基础·赋值运算符·复合赋值·自增自减
是一个Bug28 分钟前
Spring事件监听器源码深度解析
java·数据库·spring
蜂蜜黄油呀土豆33 分钟前
ThreadLocal 深度解析:它解决了什么、原理是什么、如何正确使用(含代码与实战建议)
java·并发编程·内存泄漏·threadlocal
毕设源码-郭学长37 分钟前
【开题答辩全过程】以 高校教室管理系统为例,包含答辩的问题和答案
java·spring boot
罗不丢39 分钟前
UTC,Date,LocalDate转换问题解决方法
java
Klong.k41 分钟前
谈谈session、application存储对象
java·tomcat