观察者模式实现事件驱动模型(非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);
	}
	
}
程序运行截图





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

相关文章推荐

多线程 - 你知道线程栈吗

问题 1. local 变量的压栈和出栈过程 void func1(){     int a = 0;     int b = 0; } 系统中有一个栈顶指针,每次分配和回收local 变...
  • Gykimo
  • Gykimo
  • 2013-06-20 22:48
  • 11215

发一段开启摄像头的代码

原文:http://www.donews.net/zwell/archive/2004/09/24/108219.aspx//-------------------------------------...

《炉石传说》架构设计赏析(1):游戏启动流程

向暴雪的《炉石传说》学习Unity3D游戏程序架构设计。
  • Neil3D
  • Neil3D
  • 2014-09-08 22:43
  • 35155

Libevent源码分析-----Libevent工作流程探究

之前的博文讲了很多Libevent的基础构件,现在以一个实际例子来初步探究Libevent的基本工作流程。由于还有很多Libevent的细节并没有讲所以,这里的探究还是比较简洁,例子也相当简单。

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

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

libevent源码深度剖析七

libevent源码深度剖析七——事件主循环张亮     现在我们已经初步了解了libevent的Reactor组件——event_base和事件管理框架,接下来就是libevent事件处理的中心部分...

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

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

libevent源码深度剖析六

libevent源码深度剖析六——初见事件处理框架张亮 前面已经对libevent的事件处理框架和event结构体做了描述,现在是时候剖析libevent对事件的详细处理流程了,本节将分析libeve...

C++ - static_cast | reinterpret_cast | dynamic_cast

reinterpret_cast可以转换任意一个32bit整数,包括所有的指针和整数。可以把任何整数转成指针,也可以把任何指针转成整数,以及把指针转化为任意类型的指针,威力最为强大!但不能将非32bi...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:深度学习:神经网络中的前向传播和反向传播算法推导
举报原因:
原因补充:

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