观察者模式简介

概念

观察者模式(Observer Pattern)是一种行为型设计模式,用于在对象之间建立一对多的依赖关系,当一个对象的状态发生变化时,其相关依赖对象会自动收到通知并进行相应处理。

特点

  1. 松耦合:主题和观察者之间通过抽象接口进行交互,使得它们可以独立演化而不影响彼此。
  2. 一对多关系:一个主题可以有多个观察者,并且它们之间没有直接联系。
  3. 可扩展性:可以随时增加新的观察者或删除现有观察者。

优点

  1. 解耦合:将主题与具体观察者解耦,使得它们可以独立地变化和复用。
  2. 扩展性:易于添加新的观察者以及定义新的事件类型。
  3. 实时性:实现了实时更新机制,当主题状态改变时能够即刻通知相关观察者。

缺点

  1. 过度使用可能导致性能问题和复杂度增加。
  2. 触发链问题:如果观察者之间有依赖关系,那么通知链可能会导致不可预料的结果。

适用场景

  1. 当一个对象的改变需要同时影响其他多个对象时。
  2. 当系统中存在一些对象之间的联动行为,但又希望它们之间解耦合时。

实现方式

使用自定义接口

主题和观察者都实现相应接口,在主题中维护一个观察者列表,并在状态改变时遍历通知所有观察者。

实现原理:

  1. 定义一个观察者接口(Observer),其中声明了一个更新方法(update)用于接收主题状态的改变。
  2. 定义一个主题接口(Subject),其中包括添加观察者、移除观察者和通知观察者等方法。
  3. 创建具体的主题类(ConcreteSubject),该类维护了一个观察者列表,并在状态改变时遍历通知所有注册的观察者。
  4. 创建具体的观察者类(ConcreteObserver),该类实现了更新方法,在收到主题通知时进行相应操作。

实现代码:

import java.util.ArrayList;
import java.util.List;
// 观察者接口
interface Observer {void update(String newState);
}// 主题接口
interface Subject {void registerObserver(Observer observer);void removeObserver(Observer observer);void notifyObservers();
}// 具体主题类
class ConcreteSubject implements Subject {private List<Observer> observers = new ArrayList<>();private String state;public void setState(String newState) {this.state = newState;notifyObservers();}@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(state);   }    }
}// 具体观察者类
class ConcreteObserver implements Observer {private String observerState;@Overridepublic void update(String newState) {this.observerState = newState;// 执行相应操作System.out.println("Observer state updated: " + observerState);}
}// 使用示例
public class Main {public static void main(String[] args) {ConcreteSubject subject = new ConcreteSubject();Observer observer1 = new ConcreteObserver();Observer observer2 = new ConcreteObserver();subject.registerObserver(observer1);subject.registerObserver(observer2);subject.setState("New State");}
}

上述代码中,我们定义了一个观察者接口 Observer,其中包括了一个更新方法 update。然后,我们定义了一个主题接口 Subject,其中包括注册观察者、移除观察者和通知观察者等方法。接着,我们创建了具体的主题类 ConcreteSubject,该类维护了一个观察者列表,并在状态改变时遍历通知所有注册的观察者。最后,我们创建了具体的观察者类 ConcreteObserver 实现更新方法,在收到主题通知时执行相应操作。

使用自定义接口实现方式的问题:

  1. 主题与具体观察者之间存在紧耦合关系。
  2. 观察者可能无法感知到其他已注册的新类型或特定类型。

尽管存在以上问题,使用自定义接口实现方式是观察者模式的经典实现方式,并且具有简单、直观的特点。

使用Java内置Observable类和Observer接口

主题继承Observable类并调用其方法进行状态改变通知,观察者实现Observer接口并注册到主题上。

实现原理:

  1. 创建一个具体主题类(ConcreteSubject),该类继承自Observable类。
  2. 在具体主题类中定义状态改变方法,并在该方法中调用setChanged()和notifyObservers()方法来通知所有注册的观察者。
  3. 创建一个具体观察者类(ConcreteObserver),该类实现了Observer接口,并在update()方法中定义观察者收到通知后的操作。

实现代码:

import java.util.Observable;
import java.util.Observer;// 具体主题类
class ConcreteSubject extends Observable {private String state;public void setState(String newState) {this.state = newState;setChanged();notifyObservers(state);}
}// 具体观察者类
class ConcreteObserver implements Observer {private String observerState;@Overridepublic void update(Observable o, Object arg) {if (o instanceof ConcreteSubject) {this.observerState = (String) arg;// 执行相应操作System.out.println("Observer state updated: " + observerState);}}
}// 使用示例
public class Main {public static void main(String[] args) {ConcreteSubject subject = new ConcreteSubject();Observer observer1 = new ConcreteObserver();Observer observer2 = new ConcreteObserver();subject.addObserver(observer1);subject.addObserver(observer2);subject.setState("New State");}
}

上述代码中,我们创建了一个具体主题类 ConcreteSubject,该类继承自Java内置的Observable类。在该类中,我们定义了状态改变方法 setState(),并在该方法中调用setChanged()notifyObservers()来通知所有注册的观察者。

然后,我们创建了一个具体观察者类 ConcreteObserver 实现Observer接口,并在update()方法中定义观察者收到通知后的操作。

最后,在使用示例中,我们创建了一个具体主题对象和两个具体观察者对象,并将观察者注册到主题上。当主题状态改变时,会自动通知所有注册的观察者进行相应操作。使用Java内置ObservableObserver实现方式存在以下问题:

  1. Observable是一个类而不是接口,因此无法继承其他父类。
  2. Observable被标记为已过时,虽然仍可使用但不推荐使用。
  3. 观察者只能通过实现Observer接口来实现与主题的交互。

尽管存在以上问题,使用Java内置ObservableObserver实现方式可以更方便地利用已有工具来快速实现观察者模式。

使用事件机制

通过定义事件类、监听器接口以及注册监听器等方式来实现观察者模式。当事件发生时,主题发布该事件给已注册的监听器。

实现原理:

  1. 定义一个事件类(Event),该类包含了需要传递给观察者的数据。
  2. 创建一个主题类(Subject),其中包括注册观察者、移除观察者和触发事件等方法。
  3. 创建一个具体主题类(ConcreteSubject),该类继承自主题类,在具体主题中定义了相应的业务逻辑,并在合适的时机通过调用触发事件方法来通知所有注册的观察者。
  4. 创建一个接口或抽象类作为基础,定义了处理特定类型事件的方法,然后创建具体的处理器类,实现这些方法以执行相应操作。
  5. 观察者根据所需监听的特定类型事件来实现对应接口或抽象类,并在其方法内进行相应操作。

实现代码:

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;// 事件类
class Event {private String eventType;private Map<String, Object> eventData = new HashMap<>();public Event(String eventType) {this.eventType = eventType;}public String getEventType() {return eventType;}public void addData(String key, Object value) {eventData.put(key, value);}public Object getData(String key) {return eventData.get(key);}
}// 主题接口
interface Subject {void registerObserver(Observer observer, String eventType);void removeObserver(Observer observer, String eventType);void notifyObservers(Event event);
}// 具体主题类
class ConcreteSubject implements Subject {private Map<String, List<Observer>> observersMap = new HashMap<>();@Overridepublic void registerObserver(Observer observer, String eventType) {List<Observer> observers = observersMap.getOrDefault(eventType, new ArrayList<>());observers.add(observer);observersMap.put(eventType, observers);}@Overridepublic void removeObserver(Observer observer, String eventType) {if (observersMap.containsKey(eventType)) {List<Observer> observes = observersMap.get(eventType);observes.remove(observer);if (observes.isEmpty()) { // 若没有观察者监听该事件,则从观察者列表中移除该事件类型observersMap.remove(eventType); }}}@Override   public void notifyObservers(Event event) {  if (event != null && event.getEventType() != null && observersMap.containsKey(event.getEventType())) {for (Observer observe : observersMap.get(event.getEventType())) {   observe.onEventReceived(event);  }    } }
}// 观察者接口
interface Observer {void onEventReceived(Event event);  
}// 具体观察者类1,处理特定类型事件的处理器之一
class ConcreteHandler1 implements Observer {@Override   public void onEventReceived(Event event) {    if ("eventType1".equals(event.getEventType())) {     // 执行相应操作       System.out.println("ConcreteHandler1 received Event: " + event.getData("data"));     }      } 
}// 具体观察者类2,处理特定类型事件的处理器之一
class ConcreteHandler2 implements Observer {@Override   public void onEventReceived(Event event) {    if ("eventType2".equals(event.getEventType())) {     // 执行相应操作       System.out.println("ConcreteHandler2 received Event: " + event.getData("data"));     }      }
}// 使用示例
public class Main {public static void main(String[] args) {ConcreteSubject subject = new ConcreteSubject();Observer handler1 = new ConcreteHandler1();Observer handler2 = new ConcreteHandler2();subject.registerObserver(handler1, "eventType1");subject.registerObserver(handler2, "eventType2");Event event1 = new Event("eventType1");event1.addData("data", "Event data for eventType1");Event event2 = new Event("eventType2");event2.addData("data", "Event data for eventType2");subject.notifyObservers(event1);subject.notifyObservers(event2);}
}

上述代码中,我们定义了一个事件类 Event,其中包含了需要传递给观察者的数据。然后,我们创建了主题接口 Subject 和具体主题类 ConcreteSubject 来实现注册观察者、移除观察者和通知观察者等方法。

接着,我们定义了一个观察者接口 Observer ,并创建两个具体的处理器类(例如:ConcreteHandler1  `ConcreteHandler12)来实现对不同类型事件的监听和相应操作。

在使用示例中,我们创建了一个具体主题对象和两个具体观察者对象,并将观察者注册到主题上。当主题触发特定类型事件时,会自动通知对应的处理器进行相应操作。

使用事件机制实现方式可以更加灵活地处理不同类型的事件,并使得代码结构更清晰。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/59633.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

HT for Web (Hightopo) 使用心得(6)- 3D场景环境配置(天空球,雾化,辉光,景深)

在前一篇文章《Hightopo 使用心得&#xff08;5&#xff09;- 动画的实现》中&#xff0c;我们将一个直升机模型放到了3D场景中。同时&#xff0c;还利用动画实现了让该直升机围绕山体巡逻。在这篇文章中&#xff0c;我们将对上一篇的场景进行一些环境上的丰富与美化。让场景更…

《C和指针》笔记16:赋值容易出现的误区: 赋值截短

在下面的语句中&#xff0c;认为a和x被赋予相同的值的说法是不正确的&#xff1a; a x y 3;它等价于 a (x y 3);如果x是一个字符型变量&#xff0c;那么y3的值就会被截去一段&#xff0c;以便容纳于字符类型的变量中。那么a所赋的值就是这个被截短后的值。在下面这个常…

【C语言基础】变量类型,Static关键字的使用

&#x1f4e2;&#xff1a;如果你也对机器人、人工智能感兴趣&#xff0c;看来我们志同道合✨ &#x1f4e2;&#xff1a;不妨浏览一下我的博客主页【https://blog.csdn.net/weixin_51244852】 &#x1f4e2;&#xff1a;文章若有幸对你有帮助&#xff0c;可点赞 &#x1f44d;…

Effective C++条款20——宁以pass-by-reference-to-const替换pass-by-value(设计与声明)

缺省情况下C以by value方式&#xff08;一个继承自C的方式&#xff09;传递对象至&#xff08;或来自)函数。除非你另外指定&#xff0c;否则函数参数都是以实际实参的复件&#xff08;副本&#xff09;为初值&#xff0c;而调用端所获得的亦是函数返回值的一个复件。这些复件&…

浅谈单例模式在游戏开发中的应用

前言 如果在外部想在不同的时间结点、不同的位置访问某类中的成员且想要保持访问时&#xff0c;成员地址唯一。那么可以考虑将该类声明为静态类&#xff0c;但若是成员中包含公共的数据类型&#xff0c;此时便可以考虑将该类做成一个单例。 单例模式 由于类中的数据&#x…

Leetcode 易错题整理(一)5. 7. 11. 15. 33. 34

5. 最长回文子串 给你一个字符串 s&#xff0c;找到 s 中最长的回文子串。 如果字符串的反序与原始字符串相同&#xff0c;则该字符串称为回文字符串。 示例 1&#xff1a; 输入&#xff1a;s "babad" 输出&#xff1a;"bab" 解释&#xff1a;"aba&q…

几个nlp的小任务(生成任务(摘要生成))

几个nlp的小任务生成任务——摘要生成 安装库选择模型加载数据集展示数据集数据预处理 tokenizer注意特殊的 token处理组成预处理函数调用map,对数据集进行预处理微调模型,设置参数设置数据收集器,将处理好的数据喂给模型封装测评方法将参数传给 trainer,开始训练安装库 选…

vue uniapp 同意验证码滑块验证

前言 &#xff08;vue-puzzle-vcode&#xff09; 发送验证码以及登录的时候会做验证&#xff0c;防止机刷等 效果图 一、安装依赖 npm install vue-puzzle-vcode --save二、使用步骤 1.html使用 <Vcode :show"isShow" success"onSuccess"/>2.j…

C++day7

1. #include <iostream> #include <vector> #include <fstream> using namespace std; class Stu { public:string name;int id;int age;Stu(){}Stu(string n,int i,int a):name(n),id(i),age(a){}void show(){cout << "姓名&#xff1a; "…

Jmeter 如何才能做好接口测试?

现在对测试人员的要求越来越高&#xff0c;不仅仅要做好功能测试&#xff0c;对接口测试的需求也越来越多&#xff01; 所以也越来越多的同学问&#xff0c;怎样才能做好接口测试&#xff1f; 要真正的做好接口测试&#xff0c;并且弄懂如何测试接口&#xff0c;需要从如下几…

Vue2项目练手——通用后台管理项目第一节

Vue2项目练手——通用后台管理项目 知识补充yarn和npm区别npm的缺点&#xff1a;yarn的优点 npm查看镜像和设置镜像 项目介绍项目的技术栈 项目搭建文件目录 创建路由&#xff0c;引入element-uirouter/index.jsmain.jspages/Users.vuepages/Main.vuepages/Home.vuepages/Login…

启迪未来:学乐多光屏P90引领儿童智能学习革命

在当今数字化时代&#xff0c;教育方式正经历着巨大的变革&#xff0c;智能硬件为教育领域带来了前所未有的机遇和挑战。学乐多光屏学习机作为一款创新的教育智能硬件产品&#xff0c;以其独特的特点和优势&#xff0c;引领着学习机领域的发展潮流。 1. 多功能融合&#xff1a;…

Android 基础知识

一、Activity 1、onSaveInstanceState(),onRestoreInstanceState的调用时机 onSaveInstanceState 调用时机 从最近应用中选择运行其他程序时 但用户按下Home键时 屏幕方向切换时 按下电源案件时 从当前activity启动一个新的activity时 onRestorInstanceState调用时机 只…

C中字符串转16禁止数组指令

当上位机用字符串的形式下发16进制通讯指令给到下位机时的指令解析: 方法一&#xff1a;查表法 size_t charToHex(const char *data, uint8_t *result) {size_t size 0;uint8_t i 0, j 0, k 0, n 0;char listA[] {"0123456789abcdef"};uint8_t mid[256];for …

C/C++学习——单例模式(懒汉模式与饿汉模式)

C/C学习——单例模式 一、什么是单例模式&#xff1f;二、单例模式应用三、单例模式的特点注意&#xff1a;静态成员变量的使用示例代码&#xff1a; 四、单例模式C代码示例&#xff08;饿汉模式&#xff09;五、单例模式C示例代码&#xff08;懒汉模式&#xff09; 一、什么是…

记一个有趣的bug:修改结构体中的切片不生效

问题描述&#xff1a;有一个interface类型的变量&#xff0c;把一个struct赋值给了它&#xff0c;类似下面这样 package mainimport "fmt"type ResData struct {Type stringSrcid stringSearchRes interface{} }type Data struct {name []string }func mai…

【线程同步】AQS抽象排队同步器(AbstractQueuedSynchronizer)

AQS(AbstractQueuedSynchronizer)抽象排队同步器 AbstractQueuedSynchronizer AQS就是AbstractQueuedSynchronizer类 AQS其实就是JUC包下的一个基类&#xff0c;JUC下的很多内容都是基于AQS实现了部分功能&#xff0c;比如ReentrantLock&#xff0c;ThreadPoolExecutor&#…

用变压器实现德-英语言翻译【01/8】:嵌入层

一、说明 本文是“用变压器实现德-英语言翻译”系列的第一篇文章。它引入了小规模的嵌入来建立感知系统。接下来是嵌入层的变压器使用。下面简要概述了每种方法&#xff0c;然后是德语到英语的翻译。 二、技术背景 嵌入层的目标是使模型能够详细了解单词、标记或其他输入之间的…

多模态知识学习

问题背景 海量多模态数据&#xff0c;人类认知事物也是多模态的深度学习为多模态联合学习奠定基础感知智能->认知智能多模态学习case&#xff1a;微软小冰、视频平台“只看TA”&#xff08;服务特定明星粉丝等&#xff1a;优酷、爱奇艺等&#xff09;需求&#xff1a;多模态…

实验室的服务器和本地pycharm怎么做图传

参考 远程调试 qt.qpa.xcb: could not connect to display, echo DISPLAY为空[已解决]_功夫小象的博客-CSDN博客 先安装x11 MobaXterm x11-forwarding_C--G的博客-CSDN博客 我是在容器中搞得 1&#xff0c;安装qt5 pip install PyQt5 -i https://pypi.douban.com/simple …