Java 设计模式之观察者模式:构建松耦合的事件响应系统

简介: 观察者模式是Java中常用的行为型设计模式,用于构建松耦合的事件响应系统。当一个对象状态改变时,所有依赖它的观察者将自动收到通知并更新。该模式通过抽象耦合实现发布-订阅机制,广泛应用于GUI事件处理、消息通知、数据监控等场景,具有良好的可扩展性和维护性。

Java 设计模式之观察者模式:构建松耦合的事件响应系统

在软件开发中,我们经常需要处理对象之间的依赖关系 —— 当一个对象的状态发生变化时,其他依赖它的对象需要得到通知并做出相应处理。比如微信公众号的推送机制、股票价格变动时的行情更新、UI 组件状态变化时的界面刷新等。观察者模式(Observer Pattern)正是解决这类问题的经典设计模式。

什么是观察者模式?

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

这种模式的核心是松耦合—— 被观察者不需要知道具体的观察者是谁,只需要知道观察者实现了统一的接口。这使得两者可以独立变化,互不影响。

观察者模式的核心角色

观察者模式通常包含以下四个核心角色:

  1. 被观察者(Subject):也称为主题,维护一个观察者列表,提供添加、删除观察者的方法,以及通知所有观察者的方法
  2. 观察者(Observer):定义一个更新接口,当被观察者状态变化时,被调用以进行更新
  3. 具体被观察者(Concrete Subject):实现被观察者接口,当状态发生变化时,通知所有注册的观察者
  4. 具体观察者(Concrete Observer):实现观察者接口,当收到通知时执行具体的更新操作

观察者模式的工作原理

  1. 观察者向被观察者注册自己,表明希望接收通知
  2. 被观察者维护一个观察者列表
  3. 当被观察者的状态发生变化时,遍历观察者列表,调用每个观察者的更新方法
  4. 观察者收到通知后,根据被观察者的新状态执行相应操作
  5. 观察者可以随时取消注册,不再接收通知

观察者模式的代码实现

我们以 "气象站数据发布 - 显示" 系统为例来实现观察者模式。气象站(被观察者)会定期测量温度、湿度和气压,多个显示面板(观察者)需要实时显示这些数据。

1. 观察者接口

首先定义观察者接口,所有观察者都需要实现这个接口:

// 观察者接口
public interface Observer {
   
    // 当被观察者状态变化时,调用此方法更新
    void update(float temperature, float humidity, float pressure);
}

2. 被观察者接口

定义被观察者接口,包含管理观察者和通知观察者的方法:

// 被观察者接口
public interface Subject {
   
    // 注册观察者
    void registerObserver(Observer observer);

    // 移除观察者
    void removeObserver(Observer observer);

    // 通知所有观察者
    void notifyObservers();
}

3. 具体被观察者

实现气象站作为具体被观察者:

import java.util.ArrayList;
import java.util.List;

// 气象站(具体被观察者)
public class WeatherStation implements Subject {
   
    private List<Observer> observers;
    private float temperature;  // 温度
    private float humidity;     // 湿度
    private float pressure;     // 气压

    public WeatherStation() {
   
        observers = new ArrayList<>();
    }

    @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(temperature, humidity, pressure);
        }
    }

    // 当气象数据更新时调用此方法
    public void setMeasurements(float temperature, float humidity, float pressure) {
   
        this.temperature = temperature;
        this.humidity = humidity;
        this.pressure = pressure;
        measurementsChanged();  // 数据变化,通知观察者
    }

    // 数据变化的处理方法
    private void measurementsChanged() {
   
        notifyObservers();
    }
}

4. 具体观察者

实现不同的显示面板作为具体观察者:

//  CurrentConditionsDisplay.java
// 当前天气状况面板
public class CurrentConditionsDisplay implements Observer {
   
    private float temperature;
    private float humidity;

    // 可以保存被观察者的引用,以便后续取消注册
    private Subject weatherStation;

    public CurrentConditionsDisplay(Subject weatherStation) {
   
        this.weatherStation = weatherStation;
        weatherStation.registerObserver(this);  // 注册为观察者
    }

    @Override
    public void update(float temperature, float humidity, float pressure) {
   
        this.temperature = temperature;
        this.humidity = humidity;
        display();
    }

    public void display() {
   
        System.out.println("当前天气状况:温度 " + temperature + "℃,湿度 " + humidity + "%");
    }
}

// StatisticsDisplay.java
// 天气统计面板
public class StatisticsDisplay implements Observer {
   
    private float maxTemp = 0.0f;
    private float minTemp = Float.MAX_VALUE;
    private float tempSum = 0.0f;
    private int numReadings;

    public StatisticsDisplay(Subject weatherStation) {
   
        weatherStation.registerObserver(this);
    }

    @Override
    public void update(float temperature, float humidity, float pressure) {
   
        tempSum += temperature;
        numReadings++;

        if (temperature > maxTemp) {
   
            maxTemp = temperature;
        }

        if (temperature < minTemp) {
   
            minTemp = temperature;
        }

        display();
    }

    public void display() {
   
        System.out.println("温度统计:平均 " + (tempSum / numReadings) + "℃,最高 " + maxTemp + "℃,最低 " + minTemp + "℃");
    }
}

// ForecastDisplay.java
// 天气预报面板
public class ForecastDisplay implements Observer {
   
    private float currentPressure = 1013.25f;
    private float lastPressure;

    public ForecastDisplay(Subject weatherStation) {
   
        weatherStation.registerObserver(this);
    }

    @Override
    public void update(float temperature, float humidity, float pressure) {
   
        lastPressure = currentPressure;
        currentPressure = pressure;
        display();
    }

    public void display() {
   
        System.out.print("天气预报:");
        if (currentPressure > lastPressure) {
   
            System.out.println("天气将转好!");
        } else if (currentPressure == lastPressure) {
   
            System.out.println("天气将保持不变");
        } else {
   
            System.out.println("可能会有降雨");
        }
    }
}

5. 客户端测试

public class WeatherStationDemo {
   
    public static void main(String[] args) {
   
        // 创建被观察者:气象站
        WeatherStation weatherStation = new WeatherStation();

        // 创建观察者并注册
        CurrentConditionsDisplay currentDisplay = new CurrentConditionsDisplay(weatherStation);
        StatisticsDisplay statsDisplay = new StatisticsDisplay(weatherStation);
        ForecastDisplay forecastDisplay = new ForecastDisplay(weatherStation);

        // 模拟气象数据更新
        System.out.println("\n=== 第一次测量数据 ===");
        weatherStation.setMeasurements(25.6f, 65.0f, 1012.5f);

        System.out.println("\n=== 第二次测量数据 ===");
        weatherStation.setMeasurements(26.2f, 62.0f, 1013.0f);

        System.out.println("\n=== 第三次测量数据 ===");
        weatherStation.setMeasurements(24.8f, 70.0f, 1011.0f);

        // 移除一个观察者
        weatherStation.removeObserver(forecastDisplay);
        System.out.println("\n=== 第四次测量数据(已移除天气预报面板) ===");
        weatherStation.setMeasurements(25.0f, 68.0f, 1012.0f);
    }
}

运行结果:

=== 第一次测量数据 ===
当前天气状况:温度 25.6℃,湿度 65.0%
温度统计:平均 25.6℃,最高 25.6℃,最低 25.6℃
天气预报:可能会有降雨

=== 第二次测量数据 ===
当前天气状况:温度 26.2℃,湿度 62.0%
温度统计:平均 25.9℃,最高 26.2℃,最低 25.6℃
天气预报:天气将转好!

=== 第三次测量数据 ===
当前天气状况:温度 24.8℃,湿度 70.0%
温度统计:平均 25.533333℃,最高 26.2℃,最低 24.8℃
天气预报:可能会有降雨

=== 第四次测量数据(已移除天气预报面板) ===
当前天气状况:温度 25.0℃,湿度 68.0%
温度统计:平均 25.4℃,最高 26.2℃,最低 24.8℃

Java 内置的观察者模式

Java 在java.util包中提供了内置的观察者模式实现:Observable类(被观察者)和Observer接口(观察者)。不过需要注意的是,Java 9 之后这两个类已被标记为过时(@Deprecated),推荐使用更灵活的java.beans包下的相关类或自己实现。

使用内置 API 的简单示例:

import java.util.Observable;
import java.util.Observer;

// 具体被观察者
class MyObservable extends Observable {
   
    private int state;

    public void setState(int state) {
   
        this.state = state;
        setChanged();  // 标记状态已改变
        notifyObservers(state);  // 通知观察者,可传递数据
    }
}

// 具体观察者
class MyObserver implements Observer {
   
    private String name;

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

    @Override
    public void update(Observable o, Object arg) {
   
        System.out.println(name + " 收到更新:" + arg);
    }
}

// 测试
public class BuiltInObserverDemo {
   
    public static void main(String[] args) {
   
        MyObservable observable = new MyObservable();
        observable.addObserver(new MyObserver("观察者1"));
        observable.addObserver(new MyObserver("观察者2"));

        observable.setState(100);
        observable.setState(200);
    }
}

观察者模式的优缺点

优点

  1. 松耦合:被观察者和观察者之间是抽象耦合,两者可以独立变化
  2. 可扩展性:新增观察者无需修改被观察者代码,符合开闭原则
  3. 广播通信:被观察者可以一次性通知所有观察者,高效便捷
  4. 符合单一职责原则:被观察者专注于状态管理,观察者专注于状态响应

缺点

  1. 意外更新:观察者可能会收到不需要的更新,需要在设计时处理
  2. 通知顺序问题:观察者的通知顺序不固定,可能会影响结果
  3. 内存泄漏风险:如果观察者没有正确移除,可能导致内存泄漏

观察者模式的适用场景

  1. 当一个对象的改变需要同时改变其他对象,且不知道具体有多少对象需要改变时
  2. 当一个抽象模型有两个方面,其中一个方面依赖于另一个方面时
  3. 事件处理系统:如 GUI 中的按钮点击、键盘输入等事件响应
  4. 消息通知系统:如订阅 - 发布模式、消息队列等
  5. 状态监控系统:如股票行情、传感器数据监控等

总结

观察者模式通过建立对象间的一对多依赖关系,实现了状态变化的自动通知机制,有效降低了对象间的耦合度。它在现代软件开发中应用广泛,尤其是在事件驱动、分布式系统和响应式编程中。

实现观察者模式时,需要注意合理设计被观察者和观察者的接口,确保系统的灵活性和可扩展性。同时要注意及时移除不再需要的观察者,避免内存泄漏和不必要的更新操作。

掌握观察者模式,能帮助我们设计出更灵活、更易维护的系统,特别是在处理对象间复杂依赖关系时,能显著提升代码质量。

目录
相关文章
|
6月前
|
人工智能 算法 Java
Java与AI驱动区块链:构建智能合约与去中心化AI应用
区块链技术和人工智能的融合正在开创去中心化智能应用的新纪元。本文深入探讨如何使用Java构建AI驱动的区块链应用,涵盖智能合约开发、去中心化AI模型训练与推理、数据隐私保护以及通证经济激励等核心主题。我们将完整展示从区块链基础集成、智能合约编写、AI模型上链到去中心化应用(DApp)开发的全流程,为构建下一代可信、透明的智能去中心化系统提供完整技术方案。
443 3
|
7月前
|
人工智能 缓存 监控
使用LangChain4j构建Java AI智能体:让大模型学会使用工具
AI智能体是大模型技术的重要演进方向,它使模型能够主动使用工具、与环境交互,以完成复杂任务。本文详细介绍如何在Java应用中,借助LangChain4j框架构建一个具备工具使用能力的AI智能体。我们将创建一个能够进行数学计算和实时信息查询的智能体,涵盖工具定义、智能体组装、记忆管理以及Spring Boot集成等关键步骤,并展示如何通过简单的对话界面与智能体交互。
3015 1
|
7月前
|
人工智能 Java API
构建基于Java的AI智能体:使用LangChain4j与Spring AI实现RAG应用
当大模型需要处理私有、实时的数据时,检索增强生成(RAG)技术成为了核心解决方案。本文深入探讨如何在Java生态中构建具备RAG能力的AI智能体。我们将介绍新兴的Spring AI项目与成熟的LangChain4j框架,详细演示如何从零开始构建一个能够查询私有知识库的智能问答系统。内容涵盖文档加载与分块、向量数据库集成、语义检索以及与大模型的最终合成,并提供完整的代码实现,为Java开发者开启构建复杂AI智能体的大门。
4427 58
|
6月前
|
人工智能 缓存 自然语言处理
Java与多模态AI:构建支持文本、图像和音频的智能应用
随着大模型从单一文本处理向多模态能力演进,现代AI应用需要同时处理文本、图像、音频等多种信息形式。本文深入探讨如何在Java生态中构建支持多模态AI能力的智能应用。我们将完整展示集成视觉模型、语音模型和语言模型的实践方案,涵盖从文件预处理、多模态推理到结果融合的全流程,为Java开发者打开通往下一代多模态AI应用的大门。
549 41
|
6月前
|
机器学习/深度学习 人工智能 自然语言处理
Java与生成式AI:构建内容生成与创意辅助系统
生成式AI正在重塑内容创作、软件开发和创意设计的方式。本文深入探讨如何在Java生态中构建支持文本、图像、代码等多种生成任务的创意辅助系统。我们将完整展示集成大型生成模型(如GPT、Stable Diffusion)、处理生成任务队列、优化生成结果以及构建企业级生成式AI应用的全流程,为Java开发者提供构建下一代创意辅助系统的完整技术方案。
361 10
|
6月前
|
人工智能 监控 Java
Java与AI智能体:构建自主决策与工具调用的智能系统
随着AI智能体技术的快速发展,构建能够自主理解任务、制定计划并执行复杂操作的智能系统已成为新的技术前沿。本文深入探讨如何在Java生态中构建具备工具调用、记忆管理和自主决策能力的AI智能体系统。我们将完整展示从智能体架构设计、工具生态系统、记忆机制到多智能体协作的全流程,为Java开发者提供构建下一代自主智能系统的完整技术方案。
887 4
|
6月前
|
人工智能 Java 物联网
Java与边缘AI:构建离线智能的物联网与移动应用
随着边缘计算和终端设备算力的飞速发展,AI推理正从云端向边缘端迁移。本文深入探讨如何在资源受限的边缘设备上使用Java构建离线智能应用,涵盖从模型优化、推理加速到资源管理的全流程。我们将完整展示在Android设备、嵌入式系统和IoT网关中部署轻量级AI模型的技术方案,为构建真正实时、隐私安全的边缘智能应用提供完整实践指南。
523 3
|
7月前
|
人工智能 Java API
Java AI智能体实战:使用LangChain4j构建能使用工具的AI助手
随着AI技术的发展,AI智能体(Agent)能够通过使用工具来执行复杂任务,从而大幅扩展其能力边界。本文介绍如何在Java中使用LangChain4j框架构建一个能够使用外部工具的AI智能体。我们将通过一个具体示例——一个能获取天气信息和执行数学计算的AI助手,详细讲解如何定义工具、创建智能体并处理执行流程。本文包含完整的代码示例和架构说明,帮助Java开发者快速上手AI智能体的开发。
2927 8
|
6月前
|
机器学习/深度学习 人工智能 监控
Java与AI模型部署:构建企业级模型服务与生命周期管理平台
随着企业AI模型数量的快速增长,模型部署与生命周期管理成为确保AI应用稳定运行的关键。本文深入探讨如何使用Java生态构建一个企业级的模型服务平台,实现模型的版本控制、A/B测试、灰度发布、监控与回滚。通过集成Spring Boot、Kubernetes、MLflow和监控工具,我们将展示如何构建一个高可用、可扩展的模型服务架构,为大规模AI应用提供坚实的运维基础。
511 0
|
6月前
|
机器学习/深度学习 分布式计算 Java
Java与图神经网络:构建企业级知识图谱与智能推理系统
图神经网络(GNN)作为处理非欧几里得数据的前沿技术,正成为企业知识管理和智能推理的核心引擎。本文深入探讨如何在Java生态中构建基于GNN的知识图谱系统,涵盖从图数据建模、GNN模型集成、分布式图计算到实时推理的全流程。通过具体的代码实现和架构设计,展示如何将先进的图神经网络技术融入传统Java企业应用,为构建下一代智能决策系统提供完整解决方案。
605 0

热门文章

最新文章