十四、行为型(观察者模式)

server/2024/10/20 10:34:01/

观察者模式(Observer Pattern)

概念
观察者模式(Observer Pattern)是一种行为型设计模式,它定义了对象间的一对多依赖关系,当被观察的对象(主题)状态发生改变时,所有依赖它的对象(观察者)都会收到通知并自动更新。观察者模式广泛用于事件处理系统中。


应用场景

  1. 事件驱动的系统:在GUI应用程序中,当用户触发某些事件(如按钮点击、窗口关闭等),界面上多个组件可能需要做出响应。观察者模式在此时尤为合适,可以将事件源(Subject)和处理程序(Observer)解耦。

  2. 数据变化的广播机制:当数据发生变化时,多个依赖此数据的组件或模块需要更新。例如,在股市行情系统中,当股票价格更新时,所有订阅该股票的观察者都会收到通知。

  3. 发布/订阅系统:当一个对象(发布者)需要通知多个对象(订阅者)时,观察者模式可以动态注册和通知这些订阅者,从而实现发布/订阅的机制。


注意点

  • 性能问题:如果观察者较多或者通知非常频繁,可能会影响性能,因此需要注意观察者模式的使用场景,避免不必要的通知开销。
  • 内存泄漏:要避免忘记移除不再需要的观察者,尤其是在对象生命周期较短的情况下,可能会导致内存泄漏问题。
  • 通知顺序:多个观察者同时监听同一主题时,可能需要考虑通知的顺序问题。

核心要素

  1. Subject(主题/被观察者):维护观察者列表,负责添加、移除和通知观察者。
  2. Observer(观察者):定义一个更新接口,用于接收主题状态的变更通知。
  3. ConcreteSubject(具体主题):具体的主题对象,状态发生变化时通知所有的观察者。
  4. ConcreteObserver(具体观察者):具体的观察者对象,实现更新接口并根据通知做出相应动作。

Java代码完整示例

代码示例:简单观察者模式

java">// 观察者接口
interface Observer {void update(String message);
}// 具体观察者A
class ConcreteObserverA implements Observer {private String name;public ConcreteObserverA(String name) {this.name = name;}@Overridepublic void update(String message) {System.out.println(name + " 收到通知: " + message);}
}// 具体观察者B
class ConcreteObserverB implements Observer {private String name;public ConcreteObserverB(String name) {this.name = name;}@Overridepublic void update(String message) {System.out.println(name + " 收到通知: " + message);}
}// 被观察者接口
interface Subject {void registerObserver(Observer observer);void removeObserver(Observer observer);void notifyObservers(String message);
}// 具体主题
class ConcreteSubject implements Subject {private List<Observer> observers = new ArrayList<>();@Overridepublic void registerObserver(Observer observer) {observers.add(observer);}@Overridepublic void removeObserver(Observer observer) {observers.remove(observer);}@Overridepublic void notifyObservers(String message) {for (Observer observer : observers) {observer.update(message);}}
}// 客户端代码
public class ObserverPatternDemo {public static void main(String[] args) {ConcreteSubject subject = new ConcreteSubject();Observer observer1 = new ConcreteObserverA("观察者1");Observer observer2 = new ConcreteObserverB("观察者2");subject.registerObserver(observer1);subject.registerObserver(observer2);// 通知所有观察者subject.notifyObservers("主题状态发生变化");// 移除观察者1,之后再次通知subject.removeObserver(observer1);subject.notifyObservers("第二次状态变化");}
}

输出结果

观察者1 收到通知: 主题状态发生变化
观察者2 收到通知: 主题状态发生变化
观察者2 收到通知: 第二次状态变化

各种变形用法完整示例

  1. 推模式 vs 拉模式

    • 推模式(Push):主题对象主动将变更的数据推送给所有观察者,观察者不需要主动请求数据。
    • 拉模式(Pull):观察者主动从主题对象拉取所需的数据,主题只通知有更新,但不提供具体数据。

    推模式示例(之前的代码已为推模式)

    java">// notifyObservers()方法中直接将消息推送给观察者
    subject.notifyObservers("推送的数据");
    

    拉模式示例

    java">// 观察者接口
    interface Observer {void update(Subject subject);
    }// 具体观察者
    class ConcreteObserver implements Observer {private String name;public ConcreteObserver(String name) {this.name = name;}@Overridepublic void update(Subject subject) {// 主动拉取数据if (subject instanceof ConcreteSubject) {ConcreteSubject concreteSubject = (ConcreteSubject) subject;System.out.println(name + " 拉取到数据: " + concreteSubject.getState());}}
    }// 具体主题
    class ConcreteSubject implements Subject {private List<Observer> observers = new ArrayList<>();private String state;public String getState() {return state;}public void setState(String state) {this.state = state;notifyObservers();}@Overridepublic void registerObserver(Observer observer) {observers.add(observer);}@Overridepublic void removeObserver(Observer observer) {observers.remove(observer);}@Overridepublic void notifyObservers() {for (Observer observer : observers) {observer.update(this);  // 仅通知更新,不推送数据}}
    }// 客户端代码
    public class ObserverPatternPullDemo {public static void main(String[] args) {ConcreteSubject subject = new ConcreteSubject();Observer observer = new ConcreteObserver("观察者");subject.registerObserver(observer);subject.setState("新的状态");}
    }
    

    输出结果

    观察者 拉取到数据: 新的状态
    
  2. Java内置的观察者模式实现
    Java标准库提供了java.util.Observerjava.util.Observable类,尽管它们现在被认为是过时的,但仍可以用来实现观察者模式

    代码示例

    java">import java.util.Observable;
    import java.util.Observer;// 具体主题类,继承Observable
    class ConcreteSubject extends Observable {private String state;public String getState() {return state;}public void setState(String state) {this.state = state;setChanged();  // 标记状态已经改变notifyObservers(state);  // 推送通知给所有观察者}
    }// 具体观察者类,继承Observer接口
    class ConcreteObserver implements Observer {private String name;public ConcreteObserver(String name) {this.name = name;}@Overridepublic void update(Observable o, Object arg) {System.out.println(name + " 收到更新: " + arg);}
    }public class ObserverPatternBuiltInDemo {public static void main(String[] args) {ConcreteSubject subject = new ConcreteSubject();ConcreteObserver observer1 = new ConcreteObserver("观察者1");ConcreteObserver observer2 = new ConcreteObserver("观察者2");subject.addObserver(observer1);subject.addObserver(observer2);subject.setState("主题状态已更改");}
    }
    

    输出结果

    观察者1 收到更新: 主题状态已更改
    观察者2 收到更新: 主题状态已更改
    
  3. 异步观察者模式
    如果通知的处理较为耗时,可以将通知的操作异步化,防止阻塞主题的状态变化。

    代码示例(使用线程实现异步通知)

    java">class AsyncObserver implements Observer {private String name;public AsyncObserver(String name) {this.name = name;}@Overridepublic void update(String message) {// 启动新线程进行异步处理new Thread(() -> {try {Thread.sleep(1000);  // 模拟耗时操作System.out.println(name + " 异步处理收到的通知: " + message);} catch (InterruptedException e) {e.printStackTrace();}}).start();}
    }public class AsyncObserverPatternDemo {public static void main(String[] args) {ConcreteSubject subject = new ConcreteSubject();Observer asyncObserver1 = new AsyncObserver("异步观察者1");Observer asyncObserver2 = new AsyncObserver("异步观察者2");subject.registerObserver(asyncObserver1);subject.registerObserver(asyncObserver2);subject.notifyObservers("异步通知消息");}
    }
    

    输出结果

    异步观察者1 异步处理收到的通知: 异步通知消息
    异步观察者2 异步处理收到的通知: 异步通知消息
    

通过观察者模式,系统可以轻松扩展观察者,且实现了低耦合、高可扩展性。不同的变形(推拉模式、异步处理等)还可以帮助优化性能和增强可扩展性。


http://www.ppmy.cn/server/133312.html

相关文章

STM32—旋转编码器控制直流电机(标准库)

本文使用 KY-040旋转编码器 通过TC1508A电机驱动模块来控制直流电机正转和反转&#xff08;Speed&#xff1a;0-100&#xff09;&#xff0c;代码部分基于标准库&#xff0c;使用定时器输出比较两个通道来控制PWM输出。 一、KY-040旋转编码器 下图为KY-040旋转编码器&#xf…

数据结构常考基础代码题-数组倒置

题目要求 将数组 (a1, a2, a3, ..., am, b1, b2, ..., bn) 转换成 (b1, b2, ..., bn, a1, a2, a3, ..., am)。 代码实现步骤 第一步&#xff1a;定义反转函数 根据题目中的“将数组中的元素顺序反转”&#xff0c;我们需要实现一个函数 Reverse&#xff0c;用于反转数组中从…

kafka脚本工具使用

如何定位kakfa消费端消息异常问题 查看主题查看消费者组查看消费者详情&#xff08;LAG: 消费者与最新消息的滞后程度(数字越大说明消费者处理消息的速度越慢)&#xff09; 进入docker容器&#xff0c;直接运行sh脚本即可 docker exec -it <containerName> /bin/bash或…

游戏盾真的能无视攻击吗?

在当今社会&#xff0c;网络游戏已成为人们娱乐休闲的重要组成部分。随着游戏行业的快速扩展&#xff0c;网络安全挑战也随之加剧。DDoS攻击、CC攻击等恶意手段频繁出现&#xff0c;给游戏运营商及玩家带来了重重困扰。幸运的是&#xff0c;游戏盾这一专为游戏领域设计的网络安…

C语言:c语言中‘ ‘空格与‘\0‘的区别

c语言中’ ‘空格与’\0’的区别 在C语言中&#xff0c;空格和\0是两个不同的字符&#xff0c;具有不同的作用和含义。 空格&#xff08;’ &#xff09;是一个可打印的字符&#xff08;可见字符&#xff09;&#xff0c;用于表示空白区域。它的ASCII值为32&#xff0c;主要用于…

读书读到NOBEL

最近在读陈逸鹤的《程序员的自我修养》这本书&#xff0c;里面有这么一段话&#xff1a; “远古时代的人们只能创造出用于猎捕的长矛&#xff0c;而今天借助来自各行各业人 们的智慧&#xff0c;我们可以制造出高铁、大型飞机&#xff0c;并探索宇宙。但要更进一步解决人类所面…

LeetCode 24 - 两两交换链表中的节点

题目描述 给你一个链表&#xff0c;两两交换其中相邻的节点&#xff0c;并返回交换后链表的头节点。你必须在不修改节点内部的值的情况下完成本题&#xff08;即&#xff0c;只能进行节点交换&#xff09;。 解题思路 交换链表中相邻节点的问题可以通过迭代或递归来解决。本…

沪尚茗居装修秘籍:嵌入式蒸烤箱,让厨房生活更精彩

在装修厨房时&#xff0c;选择一款合适的嵌入式蒸烤箱不仅能提升烹饪效率&#xff0c;还能为厨房增添一份现代感。沪尚茗居深知用户对厨房电器的需求&#xff0c;从实际出发&#xff0c;为用户推荐选购嵌入式蒸烤箱的实用技巧&#xff0c;让厨房生活更加美好。    首先&…