用java语言实现事件委托模式
http://blog.csdn.net/yanshujun/article/details/6494447
用java语言实现事件委托模式
事件委托模式是一个比较常用的设计模式,但是Java语言本身没有对其做一定的封装,因此实现起来有一定难度(了解原理后很简单),相比之下.NET就容易了很多。
身为一个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。扩展性很高,再来一个打篮球的学生就先写个打篮球学生类,并在测试代码中告诉放哨者一下就好,放哨者完全没有变。重用性好
PS:写文档是在太累了,是写代码的好几倍,其实应该将Event 和EventHandler抽象成接口最后,实在太晚了,留给大家发挥吧。
- 猜你在找
- 11楼 wangxiaoli888 2010-05-13 07:16发表 [回复] [引用] [举报]

- [惊讶]
- 10楼 pengchengjiji 2010-05-10 22:08发表 [回复] [引用] [举报]

- 我也在看Java设计模式
- 9楼 cjn19aijia 2010-05-10 14:51发表 [回复] [引用] [举报]

- 等段时间学java,学习了。谢谢分享
- 6楼 yanshujun 2010-04-30 21:57发表 [回复] [引用] [举报]

- <div class="quote"><span class="q"><b> 付菁(职业指导)<img src='image/group/admin.gif' border=0></b>: 感谢分享,你的文章已经推荐到学生大本营首页。</span></div>[可爱]
- 1楼 fujingselene 2010-04-28 10:58发表 [回复] [引用] [举报]

- 感谢分享,你的文章已经推荐到学生大本营首页。
用java语言实现事件委托模式的更多相关文章
- 设计模式(Java语言)- 工厂方法模式
前言 在介绍工厂方法模式之前,我们需要知道这个设计模式是什么,解决了什么样的问题?在上一篇博客 设计模式(Java语言)- 简单工厂模式 介绍了简单工厂模式,然后总结了简单工厂模式的缺点: 1.当新增 ...
- 设计模式(Java语言)- 简单工厂模式
简单工厂模式有称为静态工厂模式,属于设计模式中的创建型模式.简单工厂模式通过对外提供一个静态方法来统一为类创建实例.简单工厂模式的目的是实现类与类之间解耦,其次是客户端不需要知道这个对象是如何被穿创建 ...
- 在myeclipse中使用Java语言进行spark Standalone模式应用程序开发
一.环境配置 Myeclipse中虽然已经集成了maven插件,但是由于这个插件版本较低,建立maven project会出现错误. 解决办法:自己到官网http://maven.apache.org ...
- jdbc 加载数据库驱动如何破坏双亲委托模式
导读 通过jdbc链接数据库,是每个学习Java web 方向的人必然一开始会写的代码,虽然现在各路框架都帮大家封装好了jdbc,但是研究一下jdbc链接的套路还是很意义 术语以及相 ...
- 《JAVA设计模式》之原型模式(Prototype)
在阎宏博士的<JAVA与模式>一书中开头是这样描述原型(Prototype)模式的: 原型模式属于对象的创建模式.通过给出一个原型对象来指明所有创建的对象的类型,然后用复制这个原型对象的办 ...
- Java类加载双亲委托模式优点
启动类加载器可以抢在标准扩展类加载器之前去装载类,而标准扩展类装载器可以抢在类路径加载器之前去加载那个类,类路径装载器又可以抢在自定义类装载器之前去加载类.所以Java虚拟机先从最可信的Java核心A ...
- Java设计模式补充:回调模式、事件监听器模式、观察者模式(转)
一.回调函数 为什么首先会讲回调函数呢?因为这个是理解监听器.观察者模式的关键. 什么是回调函数 所谓的回调,用于回调的函数. 回调函数只是一个功能片段,由用户按照回调函数调用约定来实现的一个函数. ...
- 设计模式(Java语言)- 原型模式
原型模式(Prototype Pattern)也有人将原型模式称为克隆模式,是属于创造型设计模式,用于创建重复的对象,提供了一种创建对象的最佳方式.原型模式需要实现Cloneable接口,来实现对象的 ...
- vue—你必须知道的 js数据类型 前端学习 CSS 居中 事件委托和this 让js调试更简单—console AMD && CMD 模式识别课程笔记(一) web攻击 web安全之XSS JSONP && CORS css 定位 react小结
vue—你必须知道的 目录 更多总结 猛戳这里 属性与方法 语法 计算属性 特殊属性 vue 样式绑定 vue事件处理器 表单控件绑定 父子组件通信 过渡效果 vue经验总结 javascript ...
随机推荐
- SQL Syntax
1.limit 语法:限制查询记录,进行分页处理:select * from article limit 0,10;(从0号记录开始,依次取10条记录) 2.like 语法:查询指定字符串的相似匹配记 ...
- MyEclipse常用设置
中文乱码设置:1.window-->preference-->general-->content type然后在<Content Types>中展开每一颗子项,并在< ...
- RLP编码
RLP(Recursive Length Prefix, 递归长度前缀编码),是Ethereum中对象序列化的一个主要的编码方式,其目的是对任意嵌套的二进制数据的序列进行编码. RLP的目的仅仅是编码 ...
- Gradient Boost Decision Tree(GBDT)中损失函数为什么是对数形式
由于最近要经常用到XGBOOST的包,不免对相关的GBDT的原理又重新学习了一遍, 发现其中在考虑损失函数的时候,是以对数log进行度量的,囿于误差平方和函数的印象 那么为什么是对数呢?可能是下面的原 ...
- GitHub 上有哪些完整的 iOS-App 源码值得参考
作者:wjh2005链接:https://www.zhihu.com/question/28518265/answer/88750562来源:知乎著作权归作者所有,转载请联系作者获得授权. 1. Co ...
- 【汇编】mov [1000],bx:immediate operand not allowed
],bx;在目的操作数中编译器会忽略[],所以[1000]就成了立即数,出现“immediate operand not allowed”改: ;mov bx,[1000];而在源操作数中,就不会出现 ...
- uva 1658(最小费用最大流)
题意:一个带权有向图,求起点到终点的两条路径权值之和最小,且两条路径没有公共点(除起点,终点): 分析:拆点法,将u拆成u和u',u-u'容量为1,费用为0,这样就能保证每个点只用一次,起点s-s'容 ...
- DateFomrat Mask in ASPxTextbox increases width ,How to disable display of dxeErrorCell on ASPxTextBox
让ASPxTextbox显示yyyy-MM-dd, 需设置 <MaskSettings Mask="yyyy-MM-dd" />.但是会导致ASPxTextbox宽度变 ...
- 做办公用品、文具方面的 B2C 是否有前景呢?
企乐买现在正在做这方面的事,从市场角度来说需求是有的,客单价和重复购买率都还可以,但是也存在几个致命问题使得施展不开举步维艰: 1.中国特有的市场环境:在美国企业办公用品一般都是网上采购,避免灰色的东 ...
- MSSQL CharIndex()用法
CHARINDEX函数返回字符或者字符串在另一个字符串中的起始位置. CHARINDEX ( expression1 , expression2 [ , start_location ] ) expr ...