Java设计模式——观察者模式(事件监听)
最近在看Tomcat和Spring的源码,在启动的时候注册了各种Listener,事件触发的时候就执行,这里就用到了设计模式中的观察者模式。
引-GUI中的事件监听
想想以前在学Java的GUI编程的时候,就用到了事件的注册监听,然后写了一个小程序试验一下:
点击按钮触发相应的事件
public class ButtonTest extends JFrame {
ButtonTest() {
JPanel panel = new JPanel();
JButton button1 = new JButton("按钮一");
JButton button2 = new JButton("按钮二");
panel.add(button1);
panel.add(button2);
this.getContentPane().add(panel);
this.setVisible(true);
button1.addActionListener(
new ActionListener() {
public void actionPerformed(ActionEvent e) {
System.out.println("你按了按钮一");
}
});
button2.addActionListener(
new ActionListener() {
public void actionPerformed(ActionEvent e) {
System.out.println("你按了按钮二");
}
});
}
public static void main(String args[]) {
new ButtonTest();
}
}
嗯,写起来确实很简单,这是封装好的,我们只需要注册事件,并实现对应的事件响应就行了。
那么这种神奇的模式是怎么实现的呢?
以下为我的分析过程。o(╯□╰)o
首先,我们来回归本质,看看到底这种模式方便在哪儿。如果不这么做的话,我们需要再每次点击按钮的代码里面把响应的内容加上,如果有一千个事件响应,我们需要把这一千个响应都写到点击按钮这个方法里面,Oh!!MyGod!太麻烦了!
如果忘掉设计模式,我们想想这种事情应该怎么做呢?
我们需要把这一千个响应管理起来,然后在事件触发的时候,把这些响应统一调用,不就行了?问题来了...
1.如何管理起来?
2.每个响应都不一样,每次写响应的时候都需要去点击事件那里加一段,改动以前的代码,此乃编程之大忌,耦合度太高
想想回答,
1.把每一个响应都通过一个响应链来管理,触发事件时,遍历这个响应链,做出响应。
2.可以定义一个接口,在响应的地方传入该接口,调用接口的方法。这样,每次需要响应的时候,实现该接口,传入具体响应的对象,这样我们就只用关心如何响应了,不用关心如何去调用响应的办法。
晕了晕了晕了……
来看看Java中的事件处理机制是如何实现的吧!
Java中的事件处理机制
来自维基百科的结构图:

图看看就行。
场景:按下开关,灯做出的响应。
1.定义事件对象
/**
* 事件对象。继承EventObject
* Created by HuangYQ on 2016/5/31.
*/
public class SwitchEvent extends EventObject {
private String switchState; //表示开关的状态
public SwitchEvent(Switch source, String switchState) {
super(source);
this.switchState = switchState;
}
public void setSwitchState(String switchState) {
this.switchState = switchState;
}
public String getSwitchState() {
return switchState;
}
}
2.定义事件监听接口
/**
* 事件监听接口
* Created by HuangYQ on 2016/5/31.
*/
public interface SwitchListener extends EventListener {
public void handleEvent(SwitchEvent switchEvent);
}
我们可以在这里,再定义事件监听类,这些类具体实现了监听功能和事件处理功能。也可以用下面主程序中的匿名内部类来实现。原理一样。
3.定义事件源对象(具体的事件源,比如说,你点击一个button,那么button就是event source,要想使button对某些事件进行响应,你就需要注册特定的listener。这里指开关)
/**
* 电源开关
* 事件源对象,类似于Swing中的button
* Created by HuangYQ on 2016/5/31.
*/
public class Switch {
private Vector switchListenerList = new Vector();
public void addListener(Object listener) {
switchListenerList.add(listener);
}
protected void open() {
SwitchEvent switchEvent = new SwitchEvent(this, "开");
notifyListeners(switchEvent);
}
protected void close() {
SwitchEvent switchEvent = new SwitchEvent(this, "关");
notifyListeners(switchEvent);
}
private void notifyListeners(SwitchEvent switchEvent) {
Iterator iterator = switchListenerList.iterator();
while (iterator.hasNext()) {
SwitchListener switchListener = (SwitchListener) iterator.next();
switchListener.handleEvent(switchEvent);
}
}
}
4.主程序
public class SwitchMain {
public static void main(String[] args) {
Switch switchTest = new Switch();
switchTest.addListener(new SwitchListener() {
@Override
public void handleEvent(SwitchEvent switchEvent) {
//Do what ever you want !
System.out.println(switchEvent.getSwitchState());
}
});
//触发
switchTest.open();
switchTest.close();
}
}
运行:
开
关
仔细分析下程序,和我们之前YY的监听器其实大致一样,
个人觉得最难理解的就是事件对象SwitchEvent了。
问题
1.为什么要继承EventObject?不继承行不行
2.事件和事件源有什么联系,为什么还需要事件对象
答:
1.继承自EventObject只是为了规范,不实现当然可以。在handleEvent的时候,我们可以通过事件对象拿到我们需要的东西,这里面就有最基本的source,也就是这里的switch对象了,
2.事件和事件源关系并不大,事件对象里面包含source罢了,通过构造函数注入。
[参考]
设计模式学习----观察者模式(事件监听实现)
java事件处理机制(自定义事件)
Java设计模式——观察者模式(事件监听)的更多相关文章
- Java中的事件监听机制
鼠标事件监听机制的三个方面: 1.事件源对象: 事件源对象就是能够产生动作的对象.在Java语言中所有的容器组件和元素组件都是事件监听中的事件源对象.Java中根据事件的动作来区分不同的事件源对象,动 ...
- java Gui编程 事件监听机制
1. GUI编程引言 以前的学习当中,我们都使用的是命令交互方式: 例如:在DOS命令行中通过javac java命令启动程序. 软件的交互的方式: 1. 命令交互方式 图书管理系统 ...
- Java界面编程—事件监听机制
组件首先要先注册事件处理器,当用户单击组件.移动鼠标或者敲击键盘时都会产生事件(Event),一旦有时间发生,应用程序就会做出对该事件的响应,这些组件就是事件源(Event source). 接受.解 ...
- Java 中的事件监听机制
看项目代码时遇到了好多事件监听机制相关的代码.现学习一下: java事件机制包含三个部分:事件.事件监听器.事件源. 1.事件:继承自java.util.EventObject类,开发人员自己定义. ...
- Java Spring 自定义事件监听
ApplicationContext 事件 定义一个context的起动监听事件 import org.springframework.context.ApplicationListener; imp ...
- 转:java中的事件监听是怎样实现随时监听的
http://blog.csdn.net/qq369201191/article/details/51204006
- SpringBoot框架(6)--事件监听
一.场景:类与类之间的消息通信,例如创建一个对象前后做拦截,日志等等相应的事件处理. 二.事件监听步骤 (1)自定义事件继承ApplicationEvent抽象类 (2)自定义事件监听器,一般实现Ap ...
- springBoot高级:自动配置分析,事件监听,启动流程分析,监控,部署
知识点梳理 课堂讲义 02-SpringBoot自动配置-@Conditional使用 Condition是Spring4.0后引入的条件化配置接口,通过实现Condition接口可以完成有条件的加载 ...
- Java事件监听机制与观察者设计模式
一. Java事件监听机制 1. 事件监听三要素: 事件源,事件对象,事件监听器 2. 三要素之间的关系:事件源注册事件监听器后,当事件源上发生某个动作时,事件源就会调用事件监听的一个方法,并将事件对 ...
随机推荐
- Tomcat9 访问 Manager App
1.介绍 在配置好Tomcat,我们往往需要访问Tomcat的Manager以及Host Manager.就需要在tomcat-users.xml中配置用户角色来实现.在地址栏输入:localhost ...
- UIBarButtonSystemItem
typedefenum { UIBarButtonSystemItemDone, UIBarButtonSystemItemCancel, UIBarButtonSystemItemEdit, UIB ...
- java线程总结(3/5)
一.线程同步和死锁问题 异步问题: package com.horizon.action; /** * 测试同步问题 * */ public class TestSync { public stati ...
- Firefly 性能测试 通报
http://bbs.gameres.com/forum.php?mod=viewthread&tid=220516 Firefly 性能测试 主要考虑点 网络IO的并发 进程间通信压力 数据 ...
- ant-design getFieldDecorator 无法获取自定义组件的值
1.自定义或第三方的表单控件,也可以与 Form 组件一起使用.只要该组件遵循以下的约定: (1)提供受控属性 value 或其它与 valuePropName 的值同名的属性. (2)提供 onCh ...
- Can report_aeroo_ooo work with Ubuntu 13.10?
来 自 :https://www.odoo.com/forum/help-1/question/can-report-aeroo-ooo-work-with-ubuntu-13-10-34314 I ...
- 飞机3D轨迹绘制(经度-纬度-高度)
使用Python绘制 #绘制三维直线图,将飞机飞行的航迹用(经度,纬度和高度)来描述 #******************************************************** ...
- node.js express安装问题
Windows下安装npm和express 1.如果不能在线安装,可以下载npm的zip解压到本地,然后把下面的bin目录加入到path中. 2.搞定npm后,执行了npm install expre ...
- android UI 适配小节
一. 像素密度表 像素密度表 比如UE 给了三张切图分别对应: mdpi, xhdpi, xxdpi 10 * 10, 20 * 20, 30 * 30 上面的值都 ...
- python 实现元组中的的数据按照list排序, python查询mysql得到的数据是元组格式,按照list格式对他们排序
需求: 需要用echart实现软件模块的统计分析,首先是对数据库的数据查询出来,然后给数据封装成列表(list)格式,数据传到前台,在echart实现绑定数据. 因为数据已经按照从大到小的顺序显示出来 ...