设计模式——观察者模式

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

适用性

  • 当一个对象的改变需要同时改变其他对象,而且不知道具体有多少对象需要改变时。
  • 当一个对象必须通知其他对象,而并不希望指定这些对象或这些对象只知道有改变发生,而不关心如何得知这一改变时。
相关推荐
yaoxin52112321 分钟前
292. Java Stream API - 使用构建器模式创建 Stream
java·开发语言
阮松云28 分钟前
code-server 配置maven
java·linux·maven
木木木一33 分钟前
Rust学习记录--C11 编写自动化测试
java·学习·rust
bug总结36 分钟前
uniapp+动态设置顶部导航栏使用详解
java·前端·javascript
a努力。40 分钟前
字节跳动Java面试被问:一致性哈希的虚拟节点和数据迁移
java·开发语言·分布式·算法·缓存·面试·哈希算法
qq_3181215941 分钟前
互联网大厂Java面试故事:支付与金融服务微服务架构、消息队列与AI风控全流程解析
java·spring boot·redis·微服务·kafka·支付系统·金融服务
文慧的科技江湖1 小时前
重卡的充电桩一般都是多少千瓦? - 慧知开源充电桩平台
java·开发语言·开源·充电桩开源平台·慧知重卡开源充电桩平台
短剑重铸之日1 小时前
《7天学会Redis》Day 3 - 持久化机制深度解析
java·redis·后端·缓存
独自破碎E1 小时前
【前序+中序】重建二叉树
java·开发语言
LawrenceMssss1 小时前
由于创建一个完整的App涉及到多个层面(如前端、后端、数据库等),并且每种语言通常有其特定的用途(如Java/Kotlin用于Android开发,Swift/Objective-C用于iOS开发,Py
android·java·ios