java中的事件机制的参与者有3种角色:  

1.event object:就是事件产生时具体的“事件”,用于listener的相应的方法之中,作为参数,一般存在与listerner的方法之中   

2.event source:具体的接受事件的实体,比如说,你点击一个button,那么button就是event source,这样你必须使button对某些事件进行相应,你就需要注册特定的listener,比如说MouseEvent之中的MouseClicked方法,这是他就必须有了add方法   

3.event listener:具体的对监听的事件类,当有其对应的event object产生的时候,它就调用相应的方法,进行处理。在windows程序设计里边这种相应使用callback机制来实现的    



先看看jdk提供的event包:

public interface EventListener:所有事件侦听器接口必须扩展的标记接口。

public class EventObject extends Object implements Serializable所有事件状态对象都将从其派生的根类。 所有 Event 在构造时都引用了对象 "source",在逻辑上认为该对象是最初发生有关 Event 的对象。           

在Java2处理事件时,没有采用dispatchEvent()-postEvent()-handleEvent()方式,采用了监听器类,每个事件类都有相关联的监听器接口。事件从事件源到监听者的传递是通过对目标监听者对象的Java方法调用进行的。      对每个明确的事件的发生,都相应地定义一个明确的Java方法。这些方法都集中定义在事件监听者(EventListener)接口中,这个接口要继承 java.util.EventListener。
实现了事件监听者接口中一些或全部方法的类就是事件监听者。      伴随着事件的发生,相应的状态通常都封装在事件状态对象中,该对象必须继承自java.util.EventObject。事件状态对象作为单参传递给应响应该事件的监听者方法中。发出某种特定事件的事件源的标识是:遵从规定的设计格式为事件监听者定义注册方法,并接受对指定事件监听者接口实例的引用。    开始之前首先问个问题:您熟悉java.util.EventObject 和java.util.EventListener两个类以及他们已有的子类吗?
  如果你已经能够熟练使用jdk为我们提供的事件监听器,并且很熟悉MouseEvent, KeyEvent, WindowEvent等等这些jdk为我们准备好的事件,那么想必你对java的事件机制已经有所理解。但是也许你还是觉得虽然用起来没什么问题,但是原理还是有些糊涂,那么下面我们再进一步自己实现这些事件和监听器,我们把这个取名为自定义事件。   其实自定义事件在java中很有用处,我们有的时候想让自己的程序产生一个事件,但有不希望(或者不可能)用鼠标,键盘之类的输入设备进行操作,比如你写一个应用程序,在这个程序中一旦收到邮件就对邮件进行相关处理,对于“收到邮件”这个事件,jdk中就没有定义。对于这样的事件,以及对于这样的事件的监听器,我们只能自己动手完成了。

其实呢,java事件处理机制就是观察者模式,我这里详细的整理一下,在并发编程中,动不动就给某一个主题注册一个监听,然后当主题状态发生变化的时候,监听就会被触发。好了,废话不多说了,我贴出几篇代码好了。

第一步:

/**
*
*/
package test; /**
* @author LinkinPark
* @date 2015年8月31日 下午3:07:34
* @Description:
* 前面我已经整理了观察者模式,这里对照这个设计模式来整理下java事件
* 这里先写个最简单的模拟
*/
public class Util
{ public static void main(String[] args)
{
// 首先初始化一个被观察者,然后在初始化一个观察者
// 然后将观察者绑定到被观察者上面,然后使用被观察者看看观察者里面的事件有没有被触发
Observable observable = new Observable();
Observer observer = new Observer();
observable.setObserver(observer);
observable.test();//事件被成功的触发了
} } //这个模拟被观察者
class Observable
{
//绑定到被观察者上面的观察者
private Observer observer; //被观察者对外面提供的操作
public void test()
{
observer.test();
} public Observer getObserver()
{
return observer;
} public void setObserver(Observer observer)
{
this.observer = observer;
}
} // 这个模拟观察者
class Observer
{
// 这个模拟事件
public void test()
{
System.out.println("这里就是事件。。。");
}
}

第二步:

/**
*
*/
package test; import java.util.ArrayList;
import java.util.List; /**
* @author LinkinPark
* @date 2015年8月31日 下午3:07:34
* @Description:
* 现在在原来的基础上稍微复杂一点,现在不在被观察者身上只绑一个观察者了。
* 现在绑定多个观察者。
*/
public class Util
{ public static void main(String[] args)
{
// 首先初始化一个被观察者,然后在初始化2个观察者
// 然后将观察者绑定到被观察者上面,然后使用被观察者看看观察者里面的事件有没有被触发
Observable observable = new Observable();
Observer observer1 = new Observer();
Observer observer2 = new Observer();
observable.registObserver(observer1);
observable.registObserver(observer2);
//调用外部事件,看看观察者里面的事件执行了没有
observable.notifyObserver("被观察者触发了操作了呢");
} } // 这个模拟被观察者
class Observable
{
// 绑定到被观察者上面的观察者
private List<Observer> observers = new ArrayList<Observer>(); // 添加一个观察者到被观察者身上
public void registObserver(Observer observer)
{
observers.add(observer);
} // 从绑定的观察者集合中去掉当前这个
public void removeObserver(Observer observer)
{
observers.remove(observer);
} // 定义外部执行
public void notifyObserver(String str)
{
for (Observer observer : observers)
{
observer.test(str);
}
} } // 这个模拟观察者
class Observer
{
// 这个模拟事件
public void test(String str)
{
System.out.println("这里就是事件-->" + str);
}
}

第三步:

/**
*
*/
package test; import java.util.ArrayList;
import java.util.List; /**
* @author LinkinPark
* @date 2015年8月31日 下午3:07:34
* @Description:
* 现在在原来的基础上稍微复杂一点,现在不在被观察者身上只绑一个观察者了。
* 现在绑定多个观察者。现在在稍微复杂一点,观察者和被观察者都面向接口编程,里面提供多个实现
*/
public class Util
{ public static void main(String[] args)
{
// 首先初始化一个被观察者,然后在初始化2个观察者
// 然后将观察者绑定到被观察者上面,然后使用被观察者看看观察者里面的事件有没有被触发
Iobservable observable = new Observable();
Iobserver observer = new Observer();
Iobserver observer1 = new Observer1();
observable.registObserver(observer);
observable.registObserver(observer1);
// 调用被观察者的外部事件,看看观察者里面的事件执行了没有
observable.notifyObserver("被观察者触发了操作了呢");
} } // 被观察者对外提供的接口,其实这里定义成抽象类,也就是模板好一点。
interface Iobservable
{
// 绑定到被观察者上面的观察者
public List<Iobserver> observers = new ArrayList<Iobserver>(); // 添加一个观察者到被观察者身上
void registObserver(Iobserver observer); // 从绑定的观察者集合中去掉当前这个
void removeObserver(Iobserver observer); // 定义外部执行
public void notifyObserver(String str);
} // 观察者对外提供的接口。
interface Iobserver
{
// 这个模拟事件
public void test(String str);
} // 这个模拟被观察者
class Observable implements Iobservable
{ public void registObserver(Iobserver observer)
{
observers.add(observer);
System.out.println("这里添加一个被观察者-->" + observer.getClass());
} public void removeObserver(Iobserver observer)
{
observers.remove(observer);
System.out.println("这里溢出一个被观察者-->" + observer.getClass());
} public void notifyObserver(String str)
{
for (Iobserver observer : observers)
{
observer.test(str);
}
} } // 这个模拟观察者0
class Observer implements Iobserver
{
public void test(String str)
{
System.out.println("这里就是Observer事件-->" + str);
}
} // 这个模拟观察者1
class Observer1 implements Iobserver
{
public void test(String str)
{
System.out.println("这里就是Observer1事件-->" + str);
}
}

第四步:

/**
*
*/
package test; import java.util.ArrayList;
import java.util.List; /**
* @author LinkinPark
* @date 2015年8月31日 下午3:07:34
* @Description:
* 现在在原来的基础上稍微复杂一点,现在不在被观察者身上只绑一个观察者了。
* 现在绑定多个观察者。现在在稍微复杂一点,观察者和被观察者都面向接口编程,里面提供多个实现。
* 最后现在提供一个最完美的观察者模式,其实和我前面那篇博客写的差不多
*/
public class Util
{ public static void main(String[] args)
{
// 首先初始化一个被观察者,然后在初始化2个观察者
// 然后将观察者绑定到被观察者上面,然后使用被观察者看看观察者里面的事件有没有被触发
Iobservable observable = new Observable();
Iobserver observer = new Observer();
Iobserver observer1 = new Observer1();
observable.registObserver(observer);
observable.registObserver(observer1);
// 调用被观察者的外部事件,看看观察者里面的事件执行了没有
System.out.println("===========华丽丽的分割线=============");
observable.notifyObserver("LinkinPark...");
} } // 被观察者对外提供的接口,其实这里定义成抽象类,也就是模板好一点。
abstract class Iobservable
{
// 绑定到被观察者上面的观察者
public List<Iobserver> observers = new ArrayList<Iobserver>(); // 添加一个观察者到被观察者身上
abstract void registObserver(Iobserver observer); // 从绑定的观察者集合中去掉当前这个
abstract void removeObserver(Iobserver observer); // 通知给观察者,让其开始做相关处理
public void notifyObserver(String str)
{
for (Iobserver observer : observers)
{
observer.test(str);
}
} } // 观察者对外提供的接口。
interface Iobserver
{
// 这个模拟事件
public void test(String str); } // 这个模拟被观察者
class Observable extends Iobservable
{ private String name; public String getName()
{
return name;
} // 在给这个被观察者重新设值name的时候,也就说现在观察者状态发生改变了,那么就要通知给观察者,让他也来做相关的处理
public void setName(String name)
{
this.name = name;
notifyObserver(name);
} public void registObserver(Iobserver observer)
{
observers.add(observer);
System.out.println("这里添加一个被观察者-->" + observer.getClass());
} public void removeObserver(Iobserver observer)
{
observers.remove(observer);
System.out.println("这里移除一个被观察者-->" + observer.getClass());
} } // 这个模拟观察者0
class Observer implements Iobserver
{ public Observer()
{
System.out.println("Observer:观察者原来的name-->" + getName());
} private String name = "Observer"; public String getName()
{
return name;
} public void setName(String name)
{
this.name = name;
} public void test(String str)
{
System.out.println("Observer:原来的name-->" + getName());
setName(str);
System.out.println("被观察者状态改变后本地的同步name-->" + getName());
}
} // 这个模拟观察者1
class Observer1 implements Iobserver
{
public Observer1()
{
System.out.println("Observer1:观察者原来的name-->" + getName());
} private String name = "Observer1"; public String getName()
{
return name;
} public void setName(String name)
{
this.name = name;
} public void test(String str)
{
System.out.println("Observer1:原来的name-->" + getName());
setName(str);
System.out.println("被观察者状态改变后本地的同步name-->" + getName());
}
}



最后一步:java事件监听了呢。

package test;

import java.util.ArrayList;
import java.util.EventListener;
import java.util.EventObject;
import java.util.Iterator;
import java.util.List; /**
* @author LinkinPark
* @date 2015年9月1日 下午1:46:52
* @Description: java事件监听机制
*/
public class Linkin
{
// 初始化主题-->被观察者
DemoSource source = new DemoSource();
// 初始化监听-->观察者
DemoListener Listener1 = new Listener1();
DemoListener Listener2 = new Listener2(); public Linkin()
{
source.addDemoListener(Listener1).addDemoListener(Listener2).addDemoListener(new DemoListener()
{
@Override
public void demoEvent(DemoEvent demoEvent)
{
System.out.println("匿名内部类收到主题推送-->" + demoEvent.getSource().getClass());
}
});
System.out.println("主题原来的状态-->" + name);
} private String name = "nameOld"; /**
* @return the name
*/
public String getName()
{
return name;
} /**
* @param name the name to set
*/
public void setName(String name)
{
this.name = name;
System.out.println("主题现在的状态-->" + name);
source.notifyDemoEvent();
} public static void main(String[] args)
{
new Linkin().setName("nameNew");
}
} // 就是事件产生时具体的“事件”,用于listener的相应的方法之中,作为参数,一般存在与listerner的方法之中
class DemoEvent extends EventObject
{
private static final long serialVersionUID = 1L;
Object obj; public DemoEvent(Object source)
{
super(source);
obj = source;
} public Object getSource()
{
return obj;
} } // 具体的对监听的事件类,当有其对应的event object产生的时候,它就调用相应的方法,进行处理。
interface DemoListener extends EventListener
{
void demoEvent(DemoEvent demoEvent);
} // 这个相当于观察者
class Listener1 implements DemoListener
{ @Override
public void demoEvent(DemoEvent demoEvent)
{
System.out.println("Listener1监听收到了主题的推送消息" + demoEvent.getSource());
}
} // 这个相当于观察者
class Listener2 implements DemoListener
{ @Override
public void demoEvent(DemoEvent demoEvent)
{
System.out.println("Listener2监听收到了主题的推送消息" + demoEvent.getSource());
}
} // 具体的接受事件的实体,也就是观察者模式中的被观察者。比如说,你点击一个button,那么button就是event source,
// 这样你必须使button对某些事件进行相应,你就需要注册特定的listener,比如说MouseEvent之中的MouseClicked方法,这是他就必须有了add方法
class DemoSource
{
private List<Object> repository = new ArrayList<Object>();
DemoListener dl; public DemoSource()
{
} // 添加监听
public DemoSource addDemoListener(DemoListener dl)
{
repository.add(dl);
return this;
} // 状态改变被触发
public void notifyDemoEvent()
{
Iterator<Object> it = repository.iterator();
while (it.hasNext())
{
dl = (DemoListener) it.next();
dl.demoEvent(new DemoEvent(this));
}
}
}

java事件处理机制的更多相关文章

  1. 从零开始理解JAVA事件处理机制(1)

    “事件”这个词已经被滥用了.正因为“事件”的被滥用,很多人在用到事件的时候不求甚解,依样画葫芦,导致学习工作了很多年,还是不清楚什么是事件处理器.什么是事件持有者.所以,如果你对于Event这个词还是 ...

  2. 从零开始理解JAVA事件处理机制(2)

    第一节中的示例过于简单<从零开始理解JAVA事件处理机制(1)>,简单到让大家觉得这样的代码简直毫无用处.但是没办法,我们要继续写这毫无用处的代码,然后引出下一阶段真正有益的代码. 一:事 ...

  3. 从零开始理解JAVA事件处理机制(3)

    我们连续写了两小节的教师-学生的例子,必然觉得无聊死了,这样的例子我们就是玩上100遍,还是不知道该怎么写真实的代码.那从本节开始,我们开始往真实代码上面去靠拢. 事件最容易理解的例子是鼠标事件:我们 ...

  4. [转]Java事件处理机制- 事件监听器的四种实现方式

    原文来自http://stefan321.iteye.com/blog/345221 自身类作为事件监听器 外部类作为事件监听器 匿名内部类作为事件监听器 内部类作为事件监听器 自身类作为事件监听器: ...

  5. Java事件处理机制(深入理解)

    本文是关于Java事件处理机制的梳理,以及有重点的介绍一些注意点,至于基础的概念啥的不多赘述. 一.Java事件处理机制初步介绍(看图理解) 根据下图,结合生活实际,可以得知监护人可以有多个,坏人对小 ...

  6. Java事件处理机制- 事件监听器的四种实现方式

    自身类作为事件监听器 外部类作为事件监听器 匿名内部类作为事件监听器 内部类作为事件监听器 自身类作为事件监听器: import javax.swing.*; import java.awt.*; i ...

  7. java 事件处理机制:按下上下左右键控制小球的运动

    /** * 加深对事件处理机制的理解 * 通过上下左右键来控制一个小球的位置 */package com.test3;import java.awt.*;import javax.swing.*;im ...

  8. java事件处理机制(自定义事件)

    java中的事件机制的参与者有3种角色: 1.event object:事件状态对象,用于listener的相应的方法之中,作为参数,一般存在与listerner的方法之中 2.event sourc ...

  9. Java事件处理机制1

    实现一个小程序,怎样通过点击不同的按钮,让面板的背景色发生相应的变化,如图: public class Demo2 extends JFrame implements ActionListener{ ...

随机推荐

  1. Windows和Linux下换行的不同

    因为测试IM会用到一些账号,于是写了一段代码从数据库里把需要的用户名.密码和手机号都一一取了出来,然后放到NotePad++中做进一步的处理. 取用户名.手机号和密码的代码如下: public cla ...

  2. 【转】python qt(pyqt)的文件打开、文件保存、文件夹选择对话框

    import PyQt4.QtCore,PyQt4.QtGui # 获取文件路径对话框 file_name = QFileDialog.getOpenFileName(self,"open ...

  3. python3之序列化(pickle&json&shelve)

    1.pickle模块 python持久化的存储数据: python程序运行中得到了一些字符串,列表,字典等数据,想要长久的保存下来,方便以后使用,而不是简单的放入内存中关机断电就丢失数据.python ...

  4. mxnet框架样本,使用C++接口

    哇塞,好久么有跟进mxnet啦,python改版了好多好多啊,突然发现C++用起来才是最爽的. 贴一个mxnet中的C++Example中的mlp网络和实现,感觉和python对接毫无违和感.真是一级 ...

  5. MyBatis之基于XML的属性与列名映射

    上一博客主要是对单表的增删改查,比较简单,而且每个属性与table表的列都是一一对应名字也一样,今天主要学习属性与table表列名不一致的处理,主要有两种一是属性与列名不一致,二是枚举的情况,这里暂时 ...

  6. [51nod1671]货物运输

    公元2222年,l国发生了一场战争. 小Y负责领导工人运输物资. 其中有m种物资的运输方案,每种运输方案形如li,ri.表示存在一种货物从li运到ri. 这里有n个城市,第i个城市与第i+1个城市相连 ...

  7. [bzoj1997][Hnoi2010]Planar(2-sat||括号序列)

    开始填连通分量的大坑了= = 然后平面图有个性质m<=3*n-6..... 由平面图的欧拉定理n-m+r=2(r为平面图的面的个数),在极大平面图的情况可以代入得到m=3*n-6. 网上的证明( ...

  8. 解决php的sha1和java的sha1(DigestUtils.sha1Hex)产生的字符串不相等的问题

    最近对接某个第三方服务,其中对接某些api需要用到他们的签名回调,根据他们传来的get参数和apiSecret进行拼接并使用sha1加密,然后返回弄成jsonp的格式返回,出于菜鸟的本能,首先是下载了 ...

  9. 【C#】数据库脚本生成工具(二)

    年C#研发的数据库文档生成工具,给之后的工作带来了便利.近日,又针对该工具,用WinForm开发了数据库脚本生成工具-DbExcelToSQL. 下面数据库文档生成工具效果图: 感兴趣的朋友可以看下[ ...

  10. UEP-按钮控制及时间

    按钮的判断 var record = ajaxgrid.getAllRecords(); if(record.length>0){ var isPack=record[0].get(" ...