当前位置: 首页 > news >正文

java23种设计模式-观察者模式

观察者模式(Observer Pattern)学习笔记


编程相关书籍分享:https://blog.csdn.net/weixin_47763579/article/details/145855793
DeepSeek使用技巧pdf资料分享:https://blog.csdn.net/weixin_47763579/article/details/145884039


1. 模式定义

行为型设计模式,定义对象间的一种一对多依赖关系,当一个对象状态发生改变时,所有依赖它的对象都会得到通知并自动更新。又称为发布-订阅模式。

2. 适用场景

✅ 需要实现事件触发机制
✅ 存在多个对象依赖一个对象状态的场景
✅ 需要实现广播通信机制
✅ 需要解耦观察者与被观察者
✅ 需要动态增减观察者对象

3. 模式结构

notifies
«interface»
Subject
+registerObserver()
+removeObserver()
+notifyObservers()
«interface»
Observer
+update()
ConcreteSubject
-state
-observers: List
+getState()
+setState()
ConcreteObserverA
+update()
ConcreteObserverB
+update()

4. 核心角色

角色说明
Subject抽象主题(被观察者),定义注册、删除、通知观察者的接口
ConcreteSubject具体主题,维护观察者列表,状态改变时通知所有观察者
Observer抽象观察者,定义更新接口
ConcreteObserver具体观察者,实现更新逻辑

5. 代码示例

5.1 气象站监控示例

// 抽象主题
interface WeatherSubject {
    void registerObserver(WeatherObserver o);
    void removeObserver(WeatherObserver o);
    void notifyObservers();
}

// 具体主题
class WeatherStation implements WeatherSubject {
    private List<WeatherObserver> observers = new ArrayList<>();
    private float temperature;
    private float humidity;
    
    public void setMeasurements(float temp, float humidity) {
        this.temperature = temp;
        this.humidity = humidity;
        notifyObservers();
    }
    
    public void registerObserver(WeatherObserver o) {
        observers.add(o);
    }
    
    public void removeObserver(WeatherObserver o) {
        observers.remove(o);
    }
    
    public void notifyObservers() {
        for (WeatherObserver o : observers) {
            o.update(temperature, humidity);
        }
    }
}

// 抽象观察者
interface WeatherObserver {
    void update(float temp, float humidity);
}

// 具体观察者
class CurrentConditionsDisplay implements WeatherObserver {
    public void update(float temp, float humidity) {
        System.out.printf("当前天气状况:温度%.1f℃ 湿度%.1f%%\n", temp, humidity);
    }
}

class StatisticsDisplay implements WeatherObserver {
    public void update(float temp, float humidity) {
        // 实现统计逻辑
        System.out.println("更新统计数据显示...");
    }
}

// 客户端
public class Client {
    public static void main(String[] args) {
        WeatherStation station = new WeatherStation();
        WeatherObserver currentDisplay = new CurrentConditionsDisplay();
        WeatherObserver statisticsDisplay = new StatisticsDisplay();
        
        station.registerObserver(currentDisplay);
        station.registerObserver(statisticsDisplay);
        
        station.setMeasurements(25.5f, 65.0f);
        /* 输出:
           当前天气状况:温度25.5℃ 湿度65.0%
           更新统计数据显示... */
    }
}

6. 模式变种

6.1 推拉模型对比

模型类型数据传递方式特点
主题主动发送完整数据给观察者观察者被动接收,可能收到不需要的数据
观察者主动从主题获取所需数据需要时获取,减少不必要数据传输
// 拉模型实现示例
interface PullObserver {
    void update(WeatherSubject subject);
}

class PullWeatherDisplay implements PullObserver {
    public void update(WeatherSubject subject) {
        if (subject instanceof WeatherStation) {
            WeatherStation ws = (WeatherStation)subject;
            System.out.println("温度:" + ws.getTemperature());
        }
    }
}

7. 优缺点分析

✔️ 优点

  • 实现松耦合
  • 支持动态添加/删除观察者
  • 符合开闭原则
  • 支持广播通信

缺点

  • 通知顺序不可控
  • 观察者过多时影响性能
  • 循环依赖可能导致系统崩溃
  • 观察者可能收到不相关通知

8. 相关模式对比

模式目的关键区别
发布-订阅模式消息通知机制通过消息代理解耦,支持更复杂路由
中介者模式对象间交互集中控制通信,观察者模式是分布式通知
责任链模式请求传递观察者模式是单向通知机制

9. 实际应用案例

  • Java Swing的事件监听机制(ActionListener
  • Spring框架的ApplicationEventApplicationListener
  • Android的BroadcastReceiver
  • JavaBeans的PropertyChangeListener
  • Reactor编程模型
  • Vue.js的响应式系统
  • Kafka消息队列

10. 最佳实践建议

  1. 使用Java内置实现
// Java 9之前可用(已过时)
import java.util.Observable;
import java.util.Observer;

// Java 9+推荐使用PropertyChangeSupport
import java.beans.PropertyChangeListener;
import java.beans.PropertyChangeSupport;
  1. 防止内存泄漏
// 及时移除观察者
subject.addObserver(observer);
// ...
subject.deleteObserver(observer);
  1. 异步通知优化
// 使用线程池异步通知
ExecutorService executor = Executors.newCachedThreadPool();
public void notifyObservers() {
    for (Observer o : observers) {
        executor.submit(() -> o.update(data));
    }
}
  1. 使用弱引用(WeakReference):
// 防止观察者无法被垃圾回收
List<WeakReference<Observer>> weakObservers = new ArrayList<>();

public void registerObserver(Observer o) {
    weakObservers.add(new WeakReference<>(o));
}
  1. 事件过滤机制
// 添加事件类型过滤
interface EventObserver {
    void update(EventType type, Object data);
}

enum EventType { TEMP_CHANGE, HUMIDITY_CHANGE }

11. 扩展应用(Spring事件机制)

// 自定义事件
public class OrderCreatedEvent extends ApplicationEvent {
    private Order order;
    public OrderCreatedEvent(Object source, Order order) {
        super(source);
        this.order = order;
    }
    // getter...
}

// 事件发布者
@Service
class OrderService {
    @Autowired
    private ApplicationEventPublisher publisher;
    
    public void createOrder(Order order) {
        // 创建订单逻辑...
        publisher.publishEvent(new OrderCreatedEvent(this, order));
    }
}

// 事件监听者
@Component
class NotificationService {
    @EventListener
    public void handleOrderCreated(OrderCreatedEvent event) {
        // 发送通知逻辑...
    }
}

🔍 设计原则体现

  1. 开闭原则(OCP):新增观察者无需修改主题
  2. 松耦合原则:主题与观察者互相不知道对方细节
  3. 单一职责原则:主题管理状态,观察者处理通知

通过观察者模式,可以实现高效的事件通知机制,特别适合需要实现实时数据同步、事件驱动架构的场景。该模式在GUI开发、分布式系统和框架设计中应用广泛,是解耦组件关系的经典解决方案。

相关文章:

  • Spring MVC框架六:Ajax技术
  • 用 DeepSeek 打样!KubeSphere LuBan 用 3 天/3 分钟“干掉”大模型部署焦虑
  • 第五六七章
  • Element Plus: el-card的内容滚动问题
  • ES 客户端 API 二次封装思想
  • C++大整数类的设计与实现
  • 点云配准技术的演进与前沿探索:从传统算法到深度学习融合(2)
  • 【Grok3】强化学习(Reinforcement Learning, RL)复习文档
  • Python 编程题 第二节:组合数字、乘法口诀表、水仙花数、反向输出四位数、判断三角形
  • 蓝桥杯 五子棋对弈
  • AI人工智能机器学习之神经网络
  • 人工智能丨大语言模型不再高不可攀!DeepSeek开源FlashMLA,开启AI新纪元
  • 验证码介绍及生成与验证(HTML + JavaScript实现)
  • 组件注册方式、传递数据
  • 实体机器人识别虚拟环境中障碍物
  • Layui页面粘贴的方法
  • python整理文件下
  • 对 flask 框架中的全局变量 request 探究
  • SQL_优化
  • JavaScript算法-合并两个有序链表
  • 国务院安委办、应急管理部进一步调度部署“五一”假期安全防范工作
  • 神舟十九号载人飞行任务取得圆满成功
  • 张炜琳已任三明市委常委、宣传部部长
  • 牛市早报|今年第二批810亿元超长期特别国债资金下达,支持消费品以旧换新
  • 科学家为AI模型设置“防火墙”,以防止被不法分子滥用
  • 准80后湖北省财政厅副厅长徐晶华已调任襄阳市副市长