java实现事件委托模式的实例详解

java实现事件委托模式的实例详解

举例说明:

一个班级,有两类学生,A类:不学习,玩,但是玩的东西不一样,有的是做游戏,与的是看电视(有点不合理)

B类:放哨的学生,专门看老师的动向,如果老师进班了就立即通知大家。

如此就形成了一个需求,放哨的学生要通知所有玩的学生:老师来了,而不同的学生有不同的反应,有的马上把电视关闭,有的停止玩游戏。

设计的要求如下,让A类学生和B类学生完全解耦,即A类完全不知道B类的学生,却可以通知B类的学生。

代码及说明如下:

Event 类,定义了一个事件类:

package lnurd.test; 

import java.lang.reflect.Method;
import java.util.Date; 

public class Event {
 //要执行方法的对象
 private Object object;
 //要执行的方法名称
 private String methodName;
 //要执行方法的参数
 private Object[] params;
 //要执行方法的参数类型
 private Class[] paramTypes; 

 public Event(){ 

 }
 public Event(Object object,String methodName,Object...args){
  this.object=object;
  this.methodName=methodName;
  this.params=args;
  contractParamTypes(this.params);
 }
 //根据参数数组生成参数类型数组
 private void contractParamTypes(Object[] params){
  this.paramTypes=new Class[params.length];
  for(int i=0;i<params.length;i++){
   this.paramTypes[i]=params[i].getClass();
  }
 } 

 public Object getObject() {
  return object;
 }
      //若干setter getter省略
 public void setParamTypes(Class[] paramTypes) {
  this.paramTypes = paramTypes;
 }
 //执行该 对象的该方法
 public void invoke() throws Exception{
  Method method=object.getClass().getMethod(this.getMethodName(), this.getParamTypes());
  if(null==method){
   return;
  }
  method.invoke(this.getObject(), this.getParams());
 }
}

EventHandler类,若干Event类的载体,同时提供一个执行所有Event的方法

package lnurd.test; 

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

public class EventHandler {
 //是用一个List
 private List<Event> objects; 

 public EventHandler(){
  objects=new ArrayList<Event>();
 }
 //添加某个对象要执行的事件,及需要的参数
 public void addEvent(Object object,String methodName,Object...args){
  objects.add(new Event(object,methodName,args));
 }
 //通知所有的对象执行指定的事件
 public void notifyX() throws Exception{
  for(Event e : objects){
   e.invoke();
  }
 }
}

放哨的学生:这里先抽象出一个抽象类,因为放哨的人有尽职尽责的,也有马马虎虎的,

但是他们有功能的方法1。增加需要帮忙放哨的学生 2。通知所有需要放哨的学生:老师来了

package lnurd.test; 

public abstract class Notifier {
 private EventHandler eventHandler=new EventHandler(); 

 public EventHandler getEventHandler() {
  return eventHandler;
 }
 public void setEventHandler(EventHandler eventHandler) {
  this.eventHandler = eventHandler;
 }
 //增加需要帮忙放哨的学生
 public abstract void addListener(Object object,String methodName,Object...args);
 //告诉所有要帮忙放哨的学生:老师来了
 public abstract void notifyX();
}

接着是放哨人的具体实现了,这里仅实现两个

1尽职尽责的放哨人GoodNotifier

2马马虎虎的放哨人BadNotifier

package lnurd.test; 

public class GoodNotifier extends Notifier { 

 @Override
 public void addListener(Object object, String methodName, Object... args) {
  System.out.println("有新的同学委托尽职尽责的放哨人!");
  this.getEventHandler().addEvent(object, methodName, args);
 } 

 @Override
 public void notifyX() {
  System.out.println("尽职尽责的放哨人告诉所有需要帮忙的同学:老师来了");
  try{
   this.getEventHandler().notifyX();
  }catch(Exception e){
   e.printStackTrace();
  }
 } 

} 

//对于BadNotifier代码类似,不再复述.

接下来是玩游戏的学生:PlayingGameListener

package lnurd.test; 

import java.util.Date; 

public class PlayingGameListener {
 public PlayingGameListener(){
  System.out.println("我正在玩游戏 开始时间"+new Date());
 }
 public void stopPlayingGame(Date date){
  System.out.println("老师来了,快回到座位上,结束时间"+date);
 }
}

在接下来是看电视的学生WatchingTVListener

package lnurd.test; 

import java.util.Date; 

public class WatchingTVListener {
 public WatchingTVListener(){
  System.out.println("我正在看电视 "+new Date());
 }
 public void stopWatchingTV(Date date){
  System.out.println("老师来了,快关闭电视 。 结束时间"+date);
 }
}

测试代码:

//创建一个尽职尽责的放哨者
Notifier goodNotifier=new GoodNotifier(); 

//创建一个玩游戏的同学,开始玩游戏
PlayingGameListener playingGameListener=new PlayingGameListener();
//创建一个看电视的同学,开始看电视
WatchingTVListener watchingTVListener=new WatchingTVListener();
//玩游戏的同学告诉放哨的同学,老师来了告诉一下
goodNotifier.addListener(playingGameListener, "stopPlayingGame",new Date());
//看电视的同学告诉放哨的同学,老师来了告诉一下
goodNotifier.addListener(watchingTVListener, "stopWatchingTV",new Date());
try{
 //一点时间后
 Thread.sleep(1000);
}catch(Exception e){
 e.printStackTrace();
}
//老师出现,放哨的人通知所有要帮忙的同学:老师来了
goodNotifier.notifyX();

点评:

1。放哨者完全不知道做游戏者的存在,完全解耦。(当然,功劳归功于Event和EventHandler,且这两个类具有通用性)

2。老师来了后游戏者停止游戏回到座位,看电视着关闭电视。(一次通知,执行了不同类的不同方法)

3。扩展性很高,再来一个打篮球的学生就先写个打篮球学生类,并在测试代码中告诉放哨者一下就好,放哨者完全没有变。重用性好

如有疑问请留言或者到本站社区交流讨论,感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

(0)

相关推荐

  • java处理按钮点击事件的方法

    不同的事件源可以产生不同类别的事件.例如,按钮可以发送一个ActionEvent对象,而窗口可以发送WindowEvent对象. AWT时间处理机制的概要: 1. 监听器对象是一个实现了特定监听器接口(listener interface)的类的实例. 2. 事件源是一个能够注册监听器对象并发送事件对象的对象. 3. 当事件发生时,事件源将事件对象传递给所有注册的监听器. 4. 监听器对象将利用事件对象中的信息决定如何对事件做出响应. 下面是监听器的一个示例: ActionListener li

  • java用arraycopy实现多击事件

    本文实例为大家分享了java用arraycopy实现多击事件的3种方法,供大家参考,具体内容如下 1.双击事件的实现 我们规定两次点击的事件间隔在500毫秒内为双击事件,这一值可以随意限定. bt_click.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { if(startTime!=0){ long endTime = System.currentTimeMillis(); i

  • java中添加按钮并添加响应事件的方法(推荐)

    关于Java容器,面板等自行百度学一下吧 </pre><pre name="code" class="java">private Button LogInbtn = new Button("登陆"); final static JFrame buyerpagemain = new JFrame(); final Container contentPane = buyerpagemain.getContentPane();

  • 详谈Java中的事件监听机制

    鼠标事件监听机制的三个方面: 1.事件源对象: 事件源对象就是能够产生动作的对象.在Java语言中所有的容器组件和元素组件都是事件监听中的事件源对象.Java中根据事件的动作来区分不同的事件源对象,动作发生在哪个组件上,那么该组件就是事件源对象 2.事件监听方法: addMouseListener(MouseListener ml) ;该方法主要用来捕获鼠标的释放,按下,点击,进入和离开的动作:捕获到相应的动作后,交由事件处理类(实现MouseListener接口)进行处理. addAction

  • JAVA用户自定义事件监听实例代码

    JAVA用户自定义事件监听实例代码 很多介绍用户自定义事件都没有例子,或是例子不全,下面写了一个完整的例子,并写入了注释以便参考,完整的实例源代码如下: package demo; import Java.util.EventObject; /** * Title: 事件处理类,继承了事件基类 * Description: * Copyright: Copyright (c) 2005 * Company: cuijiang * @author not attributable * @versi

  • JavaFX 监听窗口关闭事件实例详解

    1.写在前面 在JavaFX的程序开发的时候,在使用多线程的时候,默认情况下在程序退出的时候,新开的线程依然在后台运行. 在这种情况下,可以监听窗口关闭事件,在里面关闭子线程. 2.具体实现的样例 package sample; import javafx.application.Application; import javafx.beans.value.ChangeListener; import javafx.beans.value.ObservableValue; import java

  • Java事件处理机制(自定义事件)实例详解

    Java事件处理机制 java中的事件机制的参与者有3种角色: 1.event object:事件状态对象,用于listener的相应的方法之中,作为参数,一般存在与listerner的方法之中 2.event source:具体的事件源,比如说,你点击一个button,那么button就是event source,要想使button对某些事件进行响应,你就需要注册特定的listener. 3.event listener:对每个明确的事件的发生,都相应地定义一个明确的Java方法.这些方法都集

  • java实现事件委托模式的实例详解

    java实现事件委托模式的实例详解 举例说明: 一个班级,有两类学生,A类:不学习,玩,但是玩的东西不一样,有的是做游戏,与的是看电视(有点不合理) B类:放哨的学生,专门看老师的动向,如果老师进班了就立即通知大家. 如此就形成了一个需求,放哨的学生要通知所有玩的学生:老师来了,而不同的学生有不同的反应,有的马上把电视关闭,有的停止玩游戏. 设计的要求如下,让A类学生和B类学生完全解耦,即A类完全不知道B类的学生,却可以通知B类的学生. 代码及说明如下: Event 类,定义了一个事件类: pa

  • java 工厂模式的实例详解

    java 工厂模式的实例详解 工厂方法中的"工厂"和我们平常理解的一样:用于生产产品. 而客户是要和产品打交道,所以工厂方法模式的意义在于把客户和产品分开,达到解耦和更灵活的目的. 一般我们有一个产品的抽象类,然后有几个具体的产品,如下: //抽象产品角色 public interface Product{ void product(); } //具体产品1 public class Pro1 implements Product{ @Override public void prod

  • Java 中桥接模式——对象结构型模式的实例详解

    Java  中桥接模式--对象结构型模式的实例详解 一.意图 将抽象部分与它的实现部分分离,使他们都可以独立的变化. 二.适用性 以下一些情况使用Bridge模式 你不希望在抽象和它的实现部分之间有一个固定的绑定关系.例如这种情况可能因为,在程序运行时刻实现部分应可以被选择或者切换. 类的抽象以及它的实现都应该可以通过生成子类的方法加以扩充.这时Bridge模式使你可以对不同的抽象接口和实现部分进行组合,并分别对他们进行扩充. 对一个抽象的实现部分的修改应对客户不产生影响,即客户代码不必重新编译

  • java开发建造者模式验证实例详解

    目录 引言 经典再现 建造者模式优点及应用场景 工厂方法模式和建造者模式区别 拓展与总结 引言 创建一个类的实例,我们通常使用类中构造函数来完成对象的初始化,如果一个对象构造过程很复杂,如果将构造过程和对象使用的过程放在一起,就显得这个类很笨重,职责也不单一,最好的解决办法就是将构造过程拿出来单独进行封装,类的使用单独封装一个类就会好很多.如:mybaits中的SqlSessionFactoryBulider和SqlSessionFactory两个类,下图为SqlSessionFactoryBu

  • Java  中桥接模式——对象结构型模式的实例详解

    Java  中桥接模式--对象结构型模式的实例详解 一.意图 将抽象部分与它的实现部分分离,使他们都可以独立的变化. 二.适用性 以下一些情况使用Bridge模式 你不希望在抽象和它的实现部分之间有一个固定的绑定关系.例如这种情况可能因为,在程序运行时刻实现部分应可以被选择或者切换. 类的抽象以及它的实现都应该可以通过生成子类的方法加以扩充.这时Bridge模式使你可以对不同的抽象接口和实现部分进行组合,并分别对他们进行扩充. 对一个抽象的实现部分的修改应对客户不产生影响,即客户代码不必重新编译

  • Java中IO流 RandomAccessFile类实例详解

    Java中IO流 RandomAccessFile类实例详解 RandomAccessFile java提供的对文件内容的访问,既可以读文件,也可以写文件. 支持随机访问文件,可以访问文件的任意位置. java文件模型,在硬盘上的文件是byte byte byte存储的,是数据的集合 打开文件,有两种模式,"rw"读写."r"只读:RandomAccessFile raf = new RandomAccessFile(file, "rw");,文

  • java 设计模式(DAO)的实例详解

    java 设计模式(DAO)的实例详解 应用场景:在Java程序中,经常需要把数据持久化,也需要获取持久化的数据,但是在进行数据持久化的过程中面临诸多问题(如:数据源不同.存储类型不同.供应商不同.访问方式不同等等),请问如何能以统一的接口进行数据持久化的操作? 其实这个我没学号(≧ ﹏ ≦).我的理解就是一个产品面向的用户不是单一的,所以我们要兼容许多情况如前面提到的数据源不同.存储类型不同.供应商不同.访问方式不同等等. ★ 解决方案 DAO的理解: 1.DAO其实是利用组合工厂模式来解决问

  • Android中mvp模式使用实例详解

    MVP 是从经典的模式MVC演变而来,它们的基本思想有相通的地方:Controller/Presenter负责逻辑的处理,Model提供数据,View负 责显示.作为一种新的模式,MVP与MVC有着一个重大的区别:在MVP中View并不直接使用Model,它们之间的通信是通过Presenter (MVC中的Controller)来进行的,所有的交互都发生在Presenter内部,而在MVC中View会从直接Model中读取数据而不是通过 Controller. 在MVC里,View是可以直接访问

  • Java设计模式之原型模式的示例详解

    目录 定义 案例 需求 方案一 方案二 对比分析 总结 定义 用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象 即实现了一个原型接口,该接口用于创建当前对象的克隆,当直接创建对象的代价比较大时,则采用这种模式 案例 需求 张三要打印100000份照片 方案一 定义照片类 /** * 照片类 * @author:liyajie * @createTime:2022/2/15 11:47 * @version:1.0 */ @Data @AllArgsConstructor publi

  • Java设计模式之状态模式State Pattern详解

    目录 概述 UML类图 状态模式与策略模式 谁决定状态转换的流向 State是接口还是抽象类 应用案例分析 状态抽象类 可以抽奖的状态 奖品发放完毕状态 发放奖品的状态 不能抽奖状态 抽奖活动(Context) 测试状态模式 概述 状态模式允许对象在内部状态改变时改变它的行为,对象看起来好像修改了它的类.这个模式将状态封装成独立的类,并将动作委托到 代表当前状态的对象,我们知道行为会随着内部状态而改变. 一个对象“看起来好像修改了它的类”是什么意思呢?从客户的视角来看:如果说你使用的对象能够完全

随机推荐