观察者模式实现事件驱动模型(非GUI事件)

原创 2016年05月07日 16:25:34

先引入四人帮之书里面关于观察者模式的一段高度概括

观察者模式——

定义对象间的一种一对多的依赖关系,当一个对象的状态发生变化时,所有依赖它的对象都得到通知并自动更新。

事件驱动模型是观察者模式的一种典型应用。该模式主要由事件源,事件对象,以及事件监听器三元素构成。以常见的点击事件为例子。按钮为事件源,相当于观察者模式中的被观察者;点击为事件对象;事件监听器,相当于观察者模式中的观察者。当事件源的属性发生变化时,所有监听该事件的事件监听器都会接收到消息并作出响应。

关于“事件”这种抽象,最直观的是在于图形界面应用里,如常见的点击、拖动事件。实际上,世间万物各种属性的变化,我们都可以称为事件。例如风停了,怪物死亡了等等。

下面以具体代码作例子,说明事件驱动模型的应用。

假设这样的业务需求:游戏服务器希望在玩家升级时触发多种效果。例如玩家升级后,各种属性都会提高,开启新的系统玩法,学习新的技能……如果不采用事件驱动模型,那么写出来的代码可能是这样——

private void handleRoleUpgrade(Object role){
		if(meetUpgradeCondition(role)){//满足升级条件
			RoleManager.getInstance().upgradeAttribution(role);//属性提升
			SkillManager.getInstance().learnNewSkill(role);//学会新技能
			//其他一堆业务
		}
	}
所有相关业务的逻辑都耦合到这里。当然,这样做也有好处,最重要的一点是业务逻辑绝对清晰。

如果使用事件驱动模型,那么在这里只需分发一个升级事件,那么该事件的所有监听器便会自动作出响应。

首先定义事件类(Event.java)

事件类以玩家昵称代表事件源,将事件源绑定在事件类,是为了程序方便。

package observer;

public class Event {
	
	private final  String userName;  	 //玩家昵称
	private final EventType evtType; 	//事件类型
	
	public Event(String userName,EventType evtType){
		this.userName = userName;
		this.evtType = evtType;
	}

	public String getUserName() {
		return userName;
	}

	public EventType getEvtType() {
		return evtType;
	}
	
}
事件类型枚举(EventType.java)
package observer;

public enum EventType {
	LEVEL_UP,		//角色升级

	;
}

具体事件——升级事件(LevelUpEvent.java)

package observer;

public class LevelUpEvent extends Event{

	public LevelUpEvent(String userName, EventType evtType) {
		super(userName, evtType);
	}

}

事件监听器接口(EventListener.java)

package observer;

public interface EventListener {

	public void handleEvent(Event event);
	
}
具体事件监听器——属性变化监听器(AttrChangeListener.java)
package observer;

/**
 *  属性变化监听器
 */
public class AttrChangeListener implements EventListener{

	@Override
	public void handleEvent(Event event) {
		System.err.println(event.getUserName()+"升级了,攻击力,防御力都将大幅提升");
	}

}
具体事件监听器——学习技能监听器(SkillListener.java)

package observer;

public class SkillListener implements EventListener{

	@Override
	public void handleEvent(Event event) {
		System.err.println(event.getUserName()+"升级了,学会剑神新技能");
	}

}
事件分发器接口,主要用于事件的注册及派发(EventDispatcher.java)
package observer;

public interface EventDispatcher {

	/**
	 *  注册事件
	 */
	public void registerEvent(EventType evtType,EventListener listener);
	
	/**
	 *  派发事件
	 */
	public void fireEvent(Event event);
	
}
具体事件分发器(CommonEventDispatcher.java)
package observer;

import java.util.HashMap;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.CopyOnWriteArraySet;

public enum CommonEventDispatcher  implements EventDispatcher{
	
	INSTANCE;	 	//采用枚举实现单例模式
	
	private final Map<EventType,Set<EventListener>> observers = new HashMap<>();
	
	@Override
	public void registerEvent(EventType evtType, EventListener listener) {
		Set<EventListener> listeners = observers.get(evtType);
		if(listeners == null){
			listeners = new CopyOnWriteArraySet<EventListener>();
			observers.put(evtType, listeners);
		}
		listeners.add(listener);
	}
	
	@Override
	public void fireEvent(Event event) {
		if(event == null){
			throw new NullPointerException("event cannot be null");
		}
		
		EventType evtType = event.getEvtType();
		Set<EventListener> listeners = observers.get(evtType);
		if(listeners != null){
			for(EventListener listener:listeners){
				try{
					listener.handleEvent(event);
				}catch(Exception e){
					e.printStackTrace();  //防止其中一个listener报异常而中断其他逻辑
				}
			}
		}
	}

}
测试代码(Test.java)
package observer;

public class Test {

	public static void main(String[] args) {
		CommonEventDispatcher dispatcher = CommonEventDispatcher.INSTANCE;
		EventListener listener1 = new AttrChangeListener();
		EventListener listener2 = new SkillListener();
		
		Event levelUpEvent = new LevelUpEvent("李逍遥",EventType.LEVEL_UP);
		dispatcher.registerEvent(EventType.LEVEL_UP, listener1);
		dispatcher.registerEvent(EventType.LEVEL_UP, listener2);
		
		dispatcher.fireEvent(levelUpEvent);
	}
	
}
程序运行截图





版权声明:本文为博主原创文章,未经博主允许不得转载。

经典软件设计模型 - 事件驱动模型

模型说明 在UI编程中,常常要对鼠标点击进行相应,首先如何获得鼠标点击呢? 方式一:创建一个线程,该线程一直循环检测是否有鼠标点击,那么这个方式有以下几个缺点: 1. CPU资源浪费,可能鼠标点击的频...
  • Gykimo
  • Gykimo
  • 2013年06月27日 15:06
  • 28841

事件驱动模型实例详解(Java篇)

    或许每个软件从业者都有从学习控制台应用程序到学习可视化编程的转变过程,控制台应用程序的优点在于可以方便的练习某个语言的语法和开发习惯(如.net和java),而可视化编程的学习又可以非常方便开...
  • frank_softworks
  • frank_softworks
  • 2007年05月29日 15:17
  • 32652

走进nginx事件驱动模型

最近在看阿里陶辉前辈写的”深入理解nginx”中的nginx的事件模块。之所以想看这块内容,是因为nginx可以处理庞大的并发连接,想看看支持其背后的事件驱动是如何构建的这篇博文我也不想贴代码什么的整...
  • Shreck66
  • Shreck66
  • 2016年01月13日 11:58
  • 2471

事件驱动模型的角度来看看 JAVA NIO

## 事件驱动模型的角度来看看 JAVA NIO ## 事件驱动模型的角度来看看 java nio,先作知识的简单铺垫, 1,阻塞非阻塞 阻塞式I/O模型: (1)等待数据准备好; (2)从内...
  • sun734274006
  • sun734274006
  • 2016年02月11日 00:51
  • 624

spring 事件驱动模型简介

spring 事件驱动模型简介
  • zheng0518
  • zheng0518
  • 2017年04月11日 15:51
  • 966

经典软件设计模型 - 事件驱动模型

模型说明 在UI编程中,常常要对鼠标点击进行相应,首先如何获得鼠标点击呢? 方式一:创建一个线程,该线程一直循环检测是否有鼠标点击,那么这个方式有以下几个缺点: 1. CPU资源浪费,可能鼠标点...
  • fanwenjieok
  • fanwenjieok
  • 2014年10月25日 12:00
  • 746

事件驱动模型

围绕如何构建一个高效稳定的网络服务器程序,本文从一个最简单的服务器模型开始,依次介绍了使用多线程的服务器模型、使用非阻塞接口的服务器模型、利用select()接口实现的基于事件驱动的服务器模型,和使用...
  • q454684431
  • q454684431
  • 2016年03月29日 10:39
  • 646

Spring3.2.6中事件驱动模型实现原理深入源码分析

Spring3.2.6中事件驱动模型实现原理深入源码分析本次学习,是在新入公司熟悉项目时候开始的。因为是做页游的项目,所以涉及到gameServer做会将游戏中的业务操作日志交给logServer处理...
  • u011066648
  • u011066648
  • 2016年03月28日 19:56
  • 1766

软件架构模式-事件驱动

没有进行架构设计的应用程序通常是紧耦合的、玻璃心,难以改变。没有头绪。如果不理解应用的各个组件的内部工作方式的话很难看清它的架构特征。关于部署和维护的问题都很难回答:架构的规模如何?程序的性能如何?程...
  • Scythe666
  • Scythe666
  • 2017年03月11日 22:22
  • 1313

异步事件模型

转载至:http://blog.chinaunix.net/uid-23242010-id-94816.html traffic server设计了一个基于事件驱动的多线程模型,通过...
  • aly1989
  • aly1989
  • 2016年07月03日 23:24
  • 430
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:观察者模式实现事件驱动模型(非GUI事件)
举报原因:
原因补充:

(最多只允许输入30个字)