最近在看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设计模式——观察者模式(事件监听)的更多相关文章

  1. Java中的事件监听机制

    鼠标事件监听机制的三个方面: 1.事件源对象: 事件源对象就是能够产生动作的对象.在Java语言中所有的容器组件和元素组件都是事件监听中的事件源对象.Java中根据事件的动作来区分不同的事件源对象,动 ...

  2. java Gui编程 事件监听机制

    1.     GUI编程引言 以前的学习当中,我们都使用的是命令交互方式: 例如:在DOS命令行中通过javac java命令启动程序. 软件的交互的方式:   1. 命令交互方式    图书管理系统 ...

  3. Java界面编程—事件监听机制

    组件首先要先注册事件处理器,当用户单击组件.移动鼠标或者敲击键盘时都会产生事件(Event),一旦有时间发生,应用程序就会做出对该事件的响应,这些组件就是事件源(Event source). 接受.解 ...

  4. Java 中的事件监听机制

    看项目代码时遇到了好多事件监听机制相关的代码.现学习一下: java事件机制包含三个部分:事件.事件监听器.事件源. 1.事件:继承自java.util.EventObject类,开发人员自己定义. ...

  5. Java Spring 自定义事件监听

    ApplicationContext 事件 定义一个context的起动监听事件 import org.springframework.context.ApplicationListener; imp ...

  6. 转:java中的事件监听是怎样实现随时监听的

    http://blog.csdn.net/qq369201191/article/details/51204006

  7. SpringBoot框架(6)--事件监听

    一.场景:类与类之间的消息通信,例如创建一个对象前后做拦截,日志等等相应的事件处理. 二.事件监听步骤 (1)自定义事件继承ApplicationEvent抽象类 (2)自定义事件监听器,一般实现Ap ...

  8. springBoot高级:自动配置分析,事件监听,启动流程分析,监控,部署

    知识点梳理 课堂讲义 02-SpringBoot自动配置-@Conditional使用 Condition是Spring4.0后引入的条件化配置接口,通过实现Condition接口可以完成有条件的加载 ...

  9. Java事件监听机制与观察者设计模式

    一. Java事件监听机制 1. 事件监听三要素: 事件源,事件对象,事件监听器 2. 三要素之间的关系:事件源注册事件监听器后,当事件源上发生某个动作时,事件源就会调用事件监听的一个方法,并将事件对 ...

随机推荐

  1. TensorFlow进阶(三)---变量的创建、初始化

    变量的的创建.初始化.保存和加载 其实变量的作用在语言中相当,都有存储一些临时值的作用或者长久存储.在Tensorflow中当训练模型时,用变量来存储和更新参数.变量包含张量(Tensor)存放于内存 ...

  2. java缓存适合使用的情况

    并非所有的情况都适合于使用二级缓存,需要根据具体情况来决定.同时可以针对某一个持久化对象配置其具体的缓存策略. 适合于使用二级缓存的情况: 1.数据不会被第三方修改 一般情况下,会被hibernate ...

  3. Linux 调度器发展简述

    引言 进程调度是操作系统的核心功能.调度器只是是调度过程中的一部分,进程调度是非常复杂的过程,需要多个系统协同工作完成.本文所关注的仅为调度器,它的主要工作是在所有 RUNNING 进程中选择最合适的 ...

  4. Linux VSFTP服务器详细配置

    Linux VSFTP服务器 1.开启服务 [root@localhost root]# service vsftpd start Starting vsftpd for vsftpd:        ...

  5. 利用github和git命令,将本地项目共享到服务器上——第二章

    附上关于git命令的第一章:https://www.cnblogs.com/mlw1814011067/p/9908856.html 六.删除服务器中的文件 1. 直接物理删除(右键,删除,或者是用b ...

  6. Android 内存泄漏分析利器——leakcanary

    LeakCanary Android 和 Java 内存泄露检测. “A small leak will sink a great ship.” - Benjamin Franklin 千里之堤, 毁 ...

  7. Android获取前台进程的方法

    概述 项目中很多场景交互非常依赖于客户端的前后景状态以及其他一些辅助信息上传,譬如当前新闻在前台(看到的是新闻界面)播放时,语音开启音乐应用,此时我们希望能看到音乐界面,并且音乐在播;而在导航应用在前 ...

  8. c++中的对象复制

    (1)this指针 this是一个隐含于每个类的成员函数的特殊指针,该指针是一个指向正在被某个成员函数操作的对象的指针. 当一个对象调用成员函数时,编译程序先将对象的地址赋给this指针,也就是说,当 ...

  9. ubuntu14.04使用root用户登录桌面,ubuntu14.04root 转

    ubuntu安装好之后,默认是不能用root用户登录桌面的,只能使用普通用户或者访客登录.怎样开启root用户登录桌面呢? 先用普通用户登录,然后切换到root用户,然后执行如下命令: vi /usr ...

  10. Windows 环境下分布式跨域Session共享

    为什么还是那句话,在网上找了N篇Session共享,但真正可以直接解决问题的还是没有找到. 一.以下为本人亲测,为防止环境不一致,对本文产生歧义,限定环境如下: 1. IIS7.0 2. Asp.ne ...