Java事件机制包括三个部分:事件、事件源、事件监听器。
1. 事件
事件一般继承自java.util.EventObject类,封装了事件源对象及跟事件相关的信息。
import java.util.EventObject;
/**
* 事件类,用于封装事件源及一些与事件相关的参数.
*/
public class CusEvent extends EventObject {
private static final long serialVersionUID = 1L;
private Object source;//事件源
public CusEvent(Object source){
super(source);
this.source = source;
}
public Object getSource() {
return source;
}
public void setSource(Object source) {
this.source = source;
}
}
2. 事件源
事件源是事件发生的地方,由于事件源的某项属性或状态发生了改变(比如BUTTON被单击、TEXTBOX的值发生改变等等)导致某项事件发生。换句话说就是生成了相应的事件对象。因为事件监听器要注册在事件源上,所以事件源类中应该要有盛装监听器的容器(List、Set等等)。
/**
* 事件源.
*/
public class EventSourceObject {
private String name;
//监听器容器
private Set<CusEventListener> listener;
public EventSourceObject(){
this.listener = new HashSet<CusEventListener>();
this.name = "defaultname";
}
//给事件源注册监听器
public void addCusListener(CusEventListener cel){
this.listener.add(cel);
}
//当事件发生时,通知注册在该事件源上的所有监听器做出相应的反应(调用回调方法)
protected void notifies(){
CusEventListener cel = null;
Iterator<CusEventListener> iterator = this.listener.iterator();
while(iterator.hasNext()){
cel = iterator.next();
cel.fireCusEvent(new CusEvent(this));
}
}
public String getName() {
return name;
}
//模拟事件触发器,当成员变量name的值发生变化时,触发事件。
public void setName(String name) {
if(!this.name.equals(name)){
this.name = name;
notifies();
}
}
}
2. 事件监听器
事件监听器实现java.util.EventListener接口,注册在事件源上,当事件源的属性或状态改变时,取得相应的监听器调用其内部的回调方法。
import java.util.EventListener;
/**
* 事件监听器,实现java.util.EventListener接口。定义回调方法,将你想要做的事
* 放到这个方法下,因为事件源发生相应的事件时会调用这个方法。
*/
public class CusEventListener implements EventListener {
//事件发生后的回调方法
public void fireCusEvent(CusEvent e){
EventSourceObjecteObject = (EventSourceObject)e.getSource();
System.out.println("My name has been changed!");
System.out.println("I got a new name,named \""+eObject.getName()+"\"");
}
}
测试类
public class MainTest {
public static void main(String[] args) {
EventSourceObject object = new EventSourceObject();
//注册监听器
object.addCusListener(new CusEventListener(){
@Override
public void fireCusEvent(CusEvent e) {
super.fireCusEvent(e);
}
});
//触发事件
object.setName("AiLu");
}
}
事件、事件源、监听器三者之间的联系
事件源-----产生----->事件------>被事件监听器发现------>进入事件处理代码
import java.awt.Frame;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;
public class DemoAction extends Frame {//窗口就是事件源
public DemoAction() {
this.setTitle("窗口关闭");
this.setLocation(400, 200);
this.setSize(360, 280);
//给窗口绑定一个事件监听器
this.addWindowListener(new WindowAdapter() {
@Override
public void windowClosing(WindowEvent e) {//事件WindowEvent
System.exit(0);//关闭窗口事件, 那么退出jvm
}
});
this.setVisible(true);
}
public static void main(String[] args) {
new DemoAction();
}
}
Java事件机与观察者模式的关系
Java事件机制抽象于设计模式中的观察者模式https://blog.youkuaiyun.com/pange1991/article/details/81187882
两者之间对比 https://blog.youkuaiyun.com/axuan_k/article/details/78803382
Java web项目中事件机制的运用
tomcat等容器启动时会解析web.xml配置。
<listener>标签就是监听器的配置。
例如Spring的监听器
<listener> <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> </listener>
ContextLoaderListener就是监听器,含有一个contextInitialized方法
public void contextInitialized(ServletContextEvent event) {
this.initWebApplicationContext(event.getServletContext());
}
ServletContextEvent是事件
ServletContext是事件源
tomcat源码中StandardContext类的listenerStart()方法是关键入口,关键代码如下
...
fireContainerEvent("beforeContextInitialized", listener);
if (noPluggabilityListeners.contains(listener)) {
listener.contextInitialized(tldEvent);
} else {
listener.contextInitialized(event);
}
fireContainerEvent("afterContextInitialized", listener);
...
tomcat在加载完ServletContext后,就会触发web.xml中配置的监听器。
可以看下《Java Web项目启动加载顺序》
其他Java事件的应用例子
- zookeeper对节点的监听也是基于Java事件机制的实现