Java监听器与观察者模式

2023-12-16

Java监听器与观察者模式

Java中的监听器(Listener)和观察者模式(Observer Pattern)都是用于处理对象间的事件通知和响应的设计模式。它们的目的是在对象之间建立一种松散的耦合,使得一个对象的状态变化可以通知到其他对象,并且其他对象能够相应地作出变化。
首先我们先用两个实例来感受一下:

观察者模式:

观察者模式是一种行为设计模式,它定义了一种一对多的依赖关系,当一个对象的状态发生变化时,所有依赖于它的对象都得到通知并被自动更新。
在这里插入图片描述

在Java中,观察者模式通常使用java.util.Observable类和java.util.Observer接口来实现。被观察的对象继承Observable类,而观察者实现Observer接口。当被观察对象的状态发生改变时,它会调用notifyObservers()方法,通知所有注册的观察者。

import java.util.Observable;
import java.util.Observer;
// 被观察的对象
class MyObservable extends Observable {
    private int data;

    public int getData() {
        return data;
    }

    public void setData(int data) {
        this.data = data;
        setChanged(); // 表示状态已改变
        notifyObservers(data); // 通知观察者
    }
}

// 观察者
class MyObserver implements Observer {
    @Override
    public void update(Observable o, Object arg) {
        if (o instanceof MyObservable) {
            System.out.println("Data changed to: " + arg);
        }
    }
}

public class ObserverPatternExample {
    public static void main(String[] args) {
        MyObservable observable = new MyObservable();
        MyObserver observer = new MyObserver();

        observable.addObserver(observer);

        observable.setData(42); // 触发通知
    }
}

这个代码比较简单,在被观察对象 MyObservable发生改变时,会通知观察者,监听此对象的观察者 MyObserver会同步做出处理。
其中有一步observable.addObserver(observer); 这个是将被观察对象让观察者检测到
其内部实现主要在Observable
源码如下所示

public class Observable {
    private boolean changed = false;
    private Vector<Observer> obs = new Vector();  //观察者列表

    public Observable() {
    }

    public synchronized void addObserver(Observer o) {  //添加观察者
        if (o == null) {
            throw new NullPointerException();
        } else {
            if (!this.obs.contains(o)) {
                this.obs.addElement(o);
            }

        }
    }
    public synchronized void deleteObserver(Observer o) {  //删除观察者
        this.obs.removeElement(o);
    }
    public void notifyObservers() {   //通知观察者
        this.notifyObservers((Object)null);
    }
    public void notifyObservers(Object arg) { //通知观察者带参执行
        Object[] arrLocal;
        synchronized(this) {
            if (!this.changed) {
                return;
            }
            arrLocal = this.obs.toArray();
            this.clearChanged();
        }
        for(int i = arrLocal.length - 1; i >= 0; --i) {
            ((Observer)arrLocal[i]).update(this, arg);
        }

    }

    public synchronized void deleteObservers() { //删除所有观察者
        this.obs.removeAllElements();
    }

    protected synchronized void setChanged() {/当改变是可以通知观察者执行
        this.changed = true;
    }

    protected synchronized void clearChanged() {//当改变是可以通知观察者不执行
        this.changed = false;
    }

    public synchronized boolean hasChanged() { //获取观察者是否可以执行
        return this.changed;
    }

    public synchronized int countObservers() {//返回当前观察者个数
        return this.obs.size();
    }
}

通过源码可以看出addObserver方法可以将观察者加入到这个被观察者的属性中,通过维护一个Vector来维护所有的观察者,观察者实现Observer接口的update方法,来执行通知的方法。

监听器模式:

监听器模式并不是一个新的设计模式,它是观察者模式在特定场景下的一种改造和应用。通常,观察者模式的主题在通知观察者时,通知中不包含任何信息。如果这个过程中携带了一些其他信息,那么主题本身就成为了事件源,而携带信息的封装类就成为了事件。此时的观察者模式,也就升级为监听器了。监听器模式是观察者模式的另一种形态。
监听器模式通常包含三个角色:事件源、事件对象、事件监听器。

在Java中,监听器模式通常通过接口和事件对象来实现。
具体实现如下:


// 监听器接口
interface MyListener {
    void onDataChanged(int newData);
}
// 发布者类
class EventSource {
    private MyListener listener;

    public void setListener(MyListener listener) {
        this.listener = listener;
    }

    public void fireEvent(int newData) {
        if (listener != null) {
            listener.onDataChanged(newData);
        }
    }
}

// 订阅者类
class MySubscriber implements MyListener {
    @Override
    public void onDataChanged(int newData) {
        System.out.println("Data changed to: " + newData);
    }
}

public class ListenerPatternExample {
    public static void main(String[] args) {
        EventSource eventSource = new EventSource();
        MySubscriber subscriber = new MySubscriber();
        eventSource.setListener(subscriber);
        eventSource.fireEvent(42); // 触发事件
    }
}

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Java监听器与观察者模式 的相关文章

  • 如何自定义BlockingQueue的阻塞行为

    我想创建一个阻塞队列 它根据自定义规则而不是队列中的项目数量来阻止生产者 例如 生产者生成一些文件并放入队列中 消费者经过一番分析后将它们转移到特定位置 对于上述场景 如果队列中的总文件大小达到某个阈值 我希望生产者等待生成新文件 如果总大
  • Hibernate OneToMany 关系是 PersistentBag 而不是 List

    我正在 javafx 中开发一个应用程序 它通过 RMI 与 EAR 连接 该 EAR 连接到 SQLServer DB 并使用 hibernate 映射 POJOS 这些 POJOS 包含双向 OneToMany 和 ManyToOne
  • JPanel透明背景和显示元素[重复]

    这个问题在这里已经有答案了 我插入一个背景图e 变成 aJPanel但一些界面元素消失了 以下 Java Swing 元素不会出现 标签标题 标签 usuario 标签 密码 按钮加速器 你能否使图像透明或元素不透明 setOpaque f
  • Spring webflow 应用程序:HTTP 302 暂时移动

    我的 java 应用程序中的每个请求都会生成另外 2 个带有 HTTP 302 错误的请求 例如 如果请求查看名为板 html 这个请求是从首页 html 我收到按以下顺序生成的 3 个请求 POST home html 302 Moved
  • 使用 jdbc 程序连接到 Open Office odb 文件

    我编写了以下代码来连接到 OpenOffice db String db C Documents and Settings hkonakanchi Desktop Test odb Class forName org hsqldb jdbc
  • 如何在 OpenAPI 3.0 中定义字节数组

    我正在将 API 从 Swagger 2 0 迁移到 OpenAPI 3 0 在 DTO 中 我有一个指定为字节数组的字段 Swagger 对 DTO 的定义 Job type object properties body type str
  • Java:检查给定日期是否在当前月份内

    我需要检查给定的日期是否在当前月份 我编写了以下代码 但 IDE 提醒我getMonth https docs oracle com javase 7 docs api java util Date html getMonth and ge
  • 如何模拟一个方面

    我目前正在使用aspectj 开发一些监控工具 因为这个工具应该是技术独立的 尽可能 所以我没有使用 Spring 进行注入 但我希望我的方面能够经过单元测试 方面示例 Aspect public class ClassLoadAspect
  • Vertx HttpClient getNow 不工作

    我的 vertx HttpClient 有问题 下面的代码显示使用 vertx 和纯 java 测试 GET Vertx vertx Vertx vertx HttpClientOptions options new HttpClientO
  • grails 上的同步块在 Windows 上有效,但在 Linux 上无效

    我有一个 grails 应用程序 它依赖于服务中的同步块 当我在 Windows 上运行它时 同步按预期工作 但当我在 ams linux 上运行时 会出现 StaleObjectStateException 该问题在以下示例中重现 cla
  • Java G1 GC 处理引用对象运行缓慢

    我已经在 J ava 上运行了计数器 它24小时工作 每秒点击通过100次左右 白天 GC 处理时间从 20 60 毫秒缓慢上升到 10000 60000 毫秒 然后下降到 20 60 毫秒 这种模式不时地重复 从 GC 日志中我发现 GC
  • 存储过程将多个表返回到 spring jdbc 模板

    我正在使用 JdbcTemplate 从 Spring DAO 类调用存储过程 我的问题是 存储过程返回多个表 有没有办法使用 Spring JdbcTemplate 访问多个表 如果我使用jdbcTemplate queryForList
  • 如何计算文件中单词的长度?爪哇

    我正在尝试编写一个代码来计算文件中特定长度的单词数 例如 How are you 会打印 Proportion of 3 letter words 100 3 words 我想计算长度为 1 2 3 4 5 6 7 8 9 10 11 12
  • RMI 服务器:rmiregistry 或 LocateRegistry.createRegistry

    对于服务器端的RMI 我们需要启动吗rmiregistry程序 或者只是调用LocateRegistry createRegistry 如果两者都可以的话 各有什么优点和缺点 他们是同一件事 rmiregistry是一个单独的程序 您可以从
  • java中的比较器链

    正在阅读Oracle 关于接口的 Java 教程 https docs oracle com javase tutorial java IandI createinterface html其中给出了一个例子Card 打牌 我试图理解接口中的
  • Java SE + Spring Data + Hibernate

    我正在尝试使用 Spring Data Hibernate 启动 Java SE 应用程序 并且到目前为止已经完成了以下操作 配置文件 Configuration PropertySource classpath hibernate pro
  • Java的hashCode可以为不同的字符串产生相同的值吗?

    使用java的哈希码函数是否可以为不同的字符串提供相同的哈希码 或者如果可能的话 其可能性的 是多少 Java 哈希码是 32 位 它散列的可能字符串的数量是无限的 所以是的 会发生冲突 百分比是没有意义的 项目 字符串 的数量是无限的 而
  • 日期时间解析异常

    解析日期时 我的代码中不断出现异常错误 日期看起来像这样 Wed May 21 00 00 00 EDT 2008 这是尝试读取它的代码 DateTimeFormatter formatter DateTimeFormatter ofPat
  • 使用 Android 的 Mobile Vision API 扫描二维码

    我跟着这个tutorial http code tutsplus com tutorials reading qr codes using the mobile vision api cms 24680关于如何构建可以扫描二维码的 Andr
  • 如何在Java中跨类共享变量,我尝试了静态不起作用

    类 Testclass1 有一个变量 有一些执行会改变变量的值 现在在同一个包中有类 Testclass2 我将如何访问 Testclass2 中变量的更新值 由 Testclass1 更新 试过这个没用 注意 Testclass1和Tes

随机推荐

  • 设置bat工作目录

    在执行bat脚本的时候 如果直接双击bat脚本 此时的工作路径一般为 C Users Administrator gt 很多时候需要将工作路径设置为bat脚本所在的目录 可以在bat脚本内设置当前工作路径为bat文件所在目录 cd d dp
  • 检查网络连通性的几种方法

    检查网络连通性的几种方法 检查网络连通性是确保计算机或设备能够与其他设备或互联网通信的重要步骤 以下是一些用于检查网络连通性的方法 Ping命令 在命令提示符 Windows 或终端 Linux macOS 中 使用ping命令 例如 在W
  • Java面试八股文及答案整理( 2023年 12月最新版,持续更新)

    一 Java 基础 1 JDK 和 JRE 有什么区别 JDK Java Development Kit 的简称 java 开发工具包 提供了 java 的开发环境和运行环境 JRE Java Runtime Environment 的简称
  • 深入理解 Android Activity 启动模式

    在 Android 应用开发中 Activity 是用户界面的核心组件 而 Activity 的启动模式则是决定应用界面如何在任务栈中交互 管理以及呈现的关键因素 正确的启动模式选择能够优化用户体验 提高应用性能 并确保应用在各种情景下都能
  • 各大厂为什么要适配鸿蒙?鸿蒙到底值不值得学

    各大厂为什么要适配鸿蒙 今天在脉脉上看到这个问题 有好多人在下面回复说什么不适配就不爱国之类的话 但是我们仔细想想 这些大厂资本家真的会被这些 不爱国 的舆论影响吗 这些大厂的公关花钱分分钟就能把舆论导向指向其他地方 不然为什么网上那么多黑
  • Centeos安装mysql

    安装mysql 检查MariaDB 因为这个会和MySQL有冲突 所以先检查一下是否有安装 查看mariadb rpm qa grep mariadb 卸载mariadb mariadb libs 5 5 44 2 el7 centos x
  • 规则引擎与商业CRM的完美邂逅:将智能决策融入商业扩展

    一 背景介绍 商业CRM系统的商机模块业务复杂 场景繁多 规则调整频繁 商机流转效率一定程度决定了销售开单的效率 如何高效配合产品侧完成业务规则调整 商机流转经历了硬编码到半配置化的优化升级 过程中遇到了一些问题 也总结了一些经验 今天来和
  • 程序员未来是不是会大量失业?近期分析分享

    程序员未来是不是会大量失业 是的 随着Al的发展恐怕就在这几年 市场环境堪忧 最近两年 也就是从2022年开始 经济下行和行业不景气的情况大家都有目共睹 互联网行业的就业情况越来越不容乐观 头部互联网公司 效益出现了不同程度的下滑 最明显的
  • go-zero开发入门-API服务开发示例

    接口定义 定义 API 接口文件 接口文件 add api 的内容如下 syntax v1 info title API 接口文件示例 desc 演示如何编写 API 接口文件 author 一见 date 2023年12月07日 vers
  • RPA提升电商运营效率!用rpa实现电商主图采集分析

    作为一名经验丰富的文案编辑 经常会面临着许多重复的工作 比如电商主图的采集与分析 通常需要我们手动打开每个商品链接 下载商品主图 然后将其整理成表格 进行分析 这个过程非常繁琐 不仅耗费大量的时间和精力 还容易出错 然而 自从我们开始使用八
  • unistd.h中定义的setsid()与fork()

    setsid 是一个UNIX系统调用 用于创建一个新的会话 session 并将当前进程设置为该会话的领头进程 session leader 通常情况下 setsid 函数用于创建守护进程 daemon 以确保它与任何终端分离 从而可以在后
  • go-zero开发入门之gateway深入研究1

    创建一个 gateway 示例 main go package main import flag fmt gateway middleware github com zeromicro go zero core conf github co
  • Android Retrofit 高级使用与原理

    Android Retrofit 高级使用与原理 简介 在 Android 开发中 网络请求是一个极为关键的部分 Retrofit 作为一个强大的网络请求库 能够简化开发流程 提供高效的网络请求能力 本文将深入介绍 Retrofit 的高级
  • Java构件技术

    文章目录 Java构件技术 构件及其在信息系统项目中的重要性 常见的Java构件技术和工具 JavaBeans Servlets EJB Spring Framework Spring框架
  • 人工智能基本常识:让深度学习技术更加人性化

    近年来 人工智能技术日臻成熟 现在 许多产品和服务都依靠人工智能技术实现自动化和智能化 因此它与我们的日常生活息息相关 无论是为我们带来各种便利的家用设备 还是我们一直在使用的产品制造方式 人工智能的影响无所不在 几乎在我们生活的方方面面推
  • AD域控环境搭建操作手册

    AD域搭建 1 准备环境 1 0 介绍 什么是域控服务器 为什么需要域 域控制器的作用 部署域服务器需要考虑几个方面
  • RS485转WiFi工业路由器在冷链物流温度监控中的应用

    随着物联网技术的不断发展和应用 冷链物流行业也迎来了新的机遇和挑战 在冷链物流中 对温度监控的要求尤为重要 因为温度是保证货物质量和安全的关键因素之一 而RS485转WiFi工业路由器则成为了实现高效 可靠的温度监控系统的重要组成部分 RS
  • go-zero 的 etcd 配置

    实现代码在 core discov config go 文件中 type EtcdConf struct Hosts string Key string ID int64 json optional User string json opt
  • go-zero开发入门-API网关鉴权开发示例

    本文是 go zero开发入门 API网关开发示例 一文的延伸 继续之前请先阅读此文 在项目根目录下创建子目录 middleware 在此目录下创建文件 auth go 内容如下 鉴权中间件 package middleware impor
  • Java监听器与观察者模式

    Java监听器与观察者模式 Java中的监听器 Listener 和观察者模式 Observer Pattern 都是用于处理对象间的事件通知和响应的设计模式 它们的目的是在对象之间建立一种松散的耦合 使得一个对象的状态变化可以通知到其他对