java事件机制(订阅/消费模型)/观察者模式 : EventObject,EventListener和Source

本文主要是介绍java事件机制(订阅/消费模型)/观察者模式 : EventObject,EventListener和Source,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

参考:

java事件机制(订阅/消费模型)

使用Java标准的java.util.EventListener实现观察者-发布者设计模式

java的事件机制一般包括三个部分:EventObjectEventListenerSource

简单示例demo:

1.EventObject : 事件

java.util.EventObject是事件状态对象的基类,它封装了事件源对象以及和事件相关的信息。
所有java的事件类都需要继承该类


import java.util.EventObject;public class MsgEvent extends EventObject {private static final long serialVersionUID = 1L;//也可以用source来传递值 这里用status来传递值private int status;public MsgEvent(Object source) {super(source);}public MsgEvent(Object source, int status) {super(source);this.status = status;}public int getStatus() {return status;}public void setStatus(int status) {this.status = status;}
}

2.EventListener : 监听事件

java.util.EventListener是一个标记接口,就是说该接口内是没有任何方法的。
所有事件监听器都需要实现该接口。
事件监听器注册在事件源上,当事件源的属性或状态改变的时候,调用相应监听器内的回调方法。

import java.util.EventListener;public interface MsgListener extends EventListener {void handleMsg(MsgEvent event) throws Exception;
}

3.Source

事件源不需要实现或继承任何接口或类,它是事件最初发生的地方。
因为事件源需要注册事件监听器,所以事件源内需要有相应的盛放事件监听器的容器。

触发动作 :


import java.util.ArrayList;
import java.util.List;public class MsgManager {static List<MsgListener> msgListeners = new ArrayList<MsgListener>();public static void addListener(MsgListener listener) {msgListeners.add(listener);}public static void sendMsg(MsgEvent event) throws Exception {notifyListeners(event);}public static void notifyListeners(MsgEvent event) throws Exception {for (MsgListener listener : msgListeners) {listener.handleMsg(event);}}public static void main(String[] args) {try {//监听器AMsgManager.addListener(new MsgListener() {public void handleMsg(MsgEvent event) throws Exception {System.out.println("MsgListenerA," + event.getStatus());}});//监听器BMsgManager.addListener(new MsgListener() {public void handleMsg(MsgEvent event) throws Exception {System.out.println("MsgListenerB," + event.getStatus());}});MsgEvent msg1 = new MsgEvent("test", 11);sendMsg(msg1);MsgEvent msg2 = new MsgEvent("test", 999);sendMsg(msg2);} catch (Exception e) {e.printStackTrace();}}
}

执行main 效果

MsgListenerA,11
MsgListenerB,11
MsgListenerA,999
MsgListenerB,999

demo2 : java.util.Observable + java.util.Observer 的使用

参考:从零开始理解JAVA事件处理机制(1)

观察者:


import java.util.Observable;
import java.util.Observer;/*** 观察者,学生*/
public class Student implements Observer {private String name;public Student(String name) {this.name = name;}@Overridepublic void update(Observable observable, Object arg) {Teacher teacher = (Teacher) observable;System.out.printf("学生%s观察到(实际是被通知)%s布置了作业《%s》 \n",this.name, teacher.getName(), arg);}}

被观察者


import java.util.ArrayList;
import java.util.List;
import java.util.Observable;/*** 被观察者,教师*/
public class Teacher extends Observable {private String name;private List<String> books;public String getName() {return this.name;}public Teacher(String name) {this.name = name;books = new ArrayList<String>();}public void setHomework(String homework) {System.out.printf("%s布置了作业%s \n", this.name, homework);books.add(homework);setChanged();notifyObservers(homework);}
}

客户端

public class Client {public static void main(String[] args) {Student student1 = new Student("张三");Student student2 = new Student("李四");Teacher teacher1 = new Teacher("王老师");teacher1.addObserver(student1);teacher1.addObserver(student2);teacher1.setHomework("事件机制第一天作业");}//王老师布置了作业事件机制第一天作业 //学生李四观察到(实际是被通知)王老师布置了作业《事件机制第一天作业》 //学生张三观察到(实际是被通知)王老师布置了作业《事件机制第一天作业》 
}

demo3 : 自定义Observable + Observer 的使用

参考:从零开始理解JAVA事件处理机制(1)

观察者:

public interface Observer {void update(Observable o);
}
class ConcreteObserver1 implements Observer {public void update(Observable o) {System.out.println("观察者1观察到" +o.getClass().getSimpleName() + "发生变化");System.out.println("观察者1做出响应");}
}
class ConcreteObserver2 implements Observer {public void update(Observable o) {System.out.println("观察者2观察到" +o.getClass().getSimpleName() + "发生变化");System.out.println("观察者2做出响应");}
}

被观察者:

import java.util.ArrayList;
import java.util.List;public class Observable {private List<Observer> observers = new ArrayList<Observer>();public void addObserver(Observer o) {observers.add(o);}public void doSomething() {System.out.println("我是被观察者,我发生变化了");// 主动去通知所有的观察者notifyObservers();}public void notifyObservers() {for (Observer observer : observers) {observer.update(this);}}
}

客户端:

public class Client {public static void main(String[] args) {Observable observable = new Observable();observable.addObserver(new ConcreteObserver1());observable.addObserver(new ConcreteObserver2());observable.doSomething();}//我是被观察者,我发生变化了//观察者1观察到Observable发生变化//观察者1做出响应//观察者2观察到Observable发生变化//观察者2做出响应
}

demo4 : EventListener + EventObject : 事件机制

参考:从零开始理解JAVA事件处理机制(2)

EventListener: 观察者接口(学生)

import java.util.EventListener;public interface HomeworkListener extends EventListener {void update(HomeworkEventObject eventObject, Object obj);
}
public class Student implements HomeworkListener {private final String name;public Student(String name) {this.name = name;}@Overridepublic void update(HomeworkEventObject eventObject, Object obj) {Teacher teacher = eventObject.getTeacher();System.out.printf("学生%s观察到(实际是被通知)%s布置了作业《%s》 \n",this.name, teacher.getName(), obj);}
}

EventObject:

java.util.EventObject是事件状态对象的基类,它封装了事件源对象以及和事件相关的信息。所有java的事件类都需要继承该类

import java.util.EventObject;public class HomeworkEventObject extends EventObject {public HomeworkEventObject(Teacher teacher) {super(teacher);}public Teacher getTeacher() {return (Teacher) super.getSource();}
}

被观察者:


import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.Set;public class Teacher {private final String name;private final List<String> homeworks;/** 教师类要维护一个自己监听器(学生)的列表,为什么?* 在观察者模式中,教师是被观察者,* 继承自java.util.Observable,Observable中含了这个列表* 现在我们没有这个列表了,所以要自己创建一个*/private final Set<HomeworkListener> homeworkListenerList;public String getName() {return this.name;}public Teacher(String name) {this.name = name;this.homeworks = new ArrayList<String>();this.homeworkListenerList = new HashSet<HomeworkListener>();}public void setHomework(String homework) {System.out.printf("%s布置了作业%s\n", this.name, homework);homeworks.add(homework);HomeworkEventObject event = new HomeworkEventObject(this);/** 在观察者模式中,我们直接调用Observable的notifyObservers来通知被观察者* 现在我们只能自己通知了~~*/for (HomeworkListener listener : homeworkListenerList) {listener.update(event, homework);}}public void addObserver(HomeworkListener homeworkListener) {homeworkListenerList.add(homeworkListener);}}

测试:

public class Client {public static void main(String[] args) {Student student1 = new Student("张三");Student student2 = new Student("李四");Teacher teacher1 = new Teacher("杨振宁");teacher1.addObserver(student1);teacher1.addObserver(student2);teacher1.setHomework("事件机制第二天作业");}}

执行main打印:

杨振宁布置了作业事件机制第二天作业
学生张三观察到(实际是被通知)杨振宁布置了作业《事件机制第二天作业》 
学生李四观察到(实际是被通知)杨振宁布置了作业《事件机制第二天作业》 

这篇关于java事件机制(订阅/消费模型)/观察者模式 : EventObject,EventListener和Source的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/526578

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

一篇文章彻底搞懂macOS如何决定java环境

《一篇文章彻底搞懂macOS如何决定java环境》MacOS作为一个功能强大的操作系统,为开发者提供了丰富的开发工具和框架,下面:本文主要介绍macOS如何决定java环境的相关资料,文中通过代码... 目录方法一:使用 which命令方法二:使用 Java_home工具(Apple 官方推荐)那问题来了,

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

Java中的.close()举例详解

《Java中的.close()举例详解》.close()方法只适用于通过window.open()打开的弹出窗口,对于浏览器的主窗口,如果没有得到用户允许是不能关闭的,:本文主要介绍Java中的.... 目录当你遇到以下三种情况时,一定要记得使用 .close():用法作用举例如何判断代码中的 input

Spring Gateway动态路由实现方案

《SpringGateway动态路由实现方案》本文主要介绍了SpringGateway动态路由实现方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随... 目录前沿何为路由RouteDefinitionRouteLocator工作流程动态路由实现尾巴前沿S