热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

Tomcat和Spring中的事件机制深入讲解

这篇文章主要给大家介绍了关于Tomcat和Spring中事件机制的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面来一起看看吧

引言

最近在看tomcat源码,源码中出现了大量事件消息,可以说整个tomcat的启动流程都可以通过事件派发机制串起来,研究透了tomcat的各种事件消息,基本上对tomcat的启动流程也就有了一个整体的认识。在这一基础上,联想到之前在看spring源码过程中也存在不少事件相关知识,于是想对这两个框架中的事件派发机制做一个简单的总结,加深理解。

事件机制原理其实比较简单,抽象来看的话,设计模式中的观察者模式可以说是最经典的事件驱动机制的体现了,观察者和被观察者就体现了事件监听和事件派发的角色。还有各种MQ,其实也是事件机制的一种体现。

理解tomcat和spring中的事件机制之前,让我们先从最基本的jdk中提供的事件机制开始说起。

JDK中的事件机制

JDK中对事件机制的各个角色提供了完善的抽象,主要包括3个角色:

EventObject(事件关注内容):事件发布时需要关注的内容。jdk中提供了EventObject接口。

EventListener(事件监听者):事件监听对象,也就是对EventObject感兴趣的对象。jdk中提供了EventListener接口。

EventSource(事件源):发布事件的对象,可以在该对象中组册EventListener,然后在特定的条件下发布EventObject给已经注册的EventListener。

事件的注册与发布,需要这三个对象协同工作,可以通过下面的例子来说明各个对象的作用:

首先是事件关注内容对象MyEventObject,实现了EventObject接口。eventName参数为具体的事件关注内容

public class MyEventObject extends EventObject {

 private String eventName ;
 
 public MyEventObject (Object source, String eventName) {
  
  super(source);
  this.setEventName(eventName);
 }

 public String getEventName() {
  return eventName;
 }

 public void setEventName(String eventName) {
  this.eventName = eventName;
 }

 private static final long serialVersiOnUID= 8374250957018011175L;
}

其次是事件监听接口MyEventListener,继承了EventListener,定义了一个myEvent接口用来发布事件,任何感兴趣的监听对象都可以实现该接口来监听。

对MyEventObject感兴趣的监听者MyEventListenerImpl,实现了MyEventListener接口,当事件发布时会触发myEvent事件并收到MyEventObject对象。

public interface MyEventListener extends EventListener {

 public void myEvent(MyEventObject eventObject);
}

public class MyEventListenerImpl implements MyEventListener {

 @Override
 public void myEvent(MyEventObject eventObject) {

  System.out.println("MyEventListenerImpl --- " + eventObject.getEventName());
 }
}

最后是事件发布源对象MyEventSource,它可以注册多个事件监听对象,任何实现了MyEventListener接口的监听对象都可以注册,内部通过一个Set来存储感兴趣的监听对象,并在合适的时机会发布消息并通知所有监听对象。

public class MyEventSource {

 private Set myEventListeners = new HashSet<>();
 
 public void addListener(MyEventListener listener){
  
  this.myEventListeners.add(listener);
 }
 
 public void removeListener(MyEventListener listener){
  
  this.myEventListeners.remove(listener);
 }
 
 public void pushEvent(){
  //dosomething
  //发布push event消息
  notifyListener(new MyEventObject(this, "push event"));
 }
 
 private void notifyListener(MyEventObject eventObject){
  
  for (MyEventListener myEventListener : myEventListeners) {
   myEventListener.myEvent(eventObject);
  }
 }
}

之后可以通过一个启动类来注册并触发事件:

public static void main(String[] args) {
 
 MyEventSource myEventSource = new MyEventSource();
 
 MyEventListenerImpl myEventListenerImpl = new MyEventListenerImpl();
 
 myEventSource.addListener(myEventListenerImpl);
 
 myEventSource.pushEvent();
}

MyEventObject定义了感兴趣的内容,MyEventListenerImpl是对MyEventObject感兴趣的监听者,MyEventSource会发布MyEventObject给所有组册的监听者,最后通过一个main来启动整个流程。

明白了jdk中对事件机制的定义,再来看看tomcat和spring中的事件机制。

Tomcat的事件机制

tomcat的事件机制也离不开EventObject、EventListener以及EventSource三个对象,只不过在此基础上提供了更加抽象和便捷的操作。这里我挑选tomcat的生命周期接口对象Lifecycle来讲解整个事件发布流程:

首先还是EventObject对象LifecycleEvent,这里只列出了核心代码。它的主要参数是Lifecycle,Lifecycle中定义了tomcat各个阶段的名称:before_init、after_init、start等等,是事件监听者感兴趣的对象。

public final class LifecycleEvent extends EventObject {

 //......

 public LifecycleEvent(Lifecycle lifecycle, String type, Object data) {

  super(lifecycle);
  this.type = type;
  this.data = data;
 }

 //......
}

public interface Lifecycle {
 /**
  * The LifecycleEvent type for the "component after init" event.
  */
 public static final String BEFORE_INIT_EVENT = "before_init";

 /**
  * The LifecycleEvent type for the "component after init" event.
  */
 public static final String AFTER_INIT_EVENT = "after_init";

 /**
  * The LifecycleEvent type for the "component start" event.
  */
 public static final String START_EVENT = "start";

 //......
}

事件监听接口LifecycleListener,定义了lifecycleEvent方法用来传递监听者感兴趣的LifecycleEvent对象,监听者使用LifecycleEvent参数用来在tomcat的各个阶段处理进行相应处理。这些感兴趣的对象包括下面这些类:

这里使用ContextConfig类为例,可以看到它实现了LifecycleListener接口。这个类在解析server.xml的时候用来监听StandardContext的各个阶段的事件,并做出相应处理:

public interface LifecycleListener {

 public void lifecycleEvent(LifecycleEvent event);
}

public class ContextConfig implements LifecycleListener {
 
 //......
 @Override
 public void lifecycleEvent(LifecycleEvent event) {

  // Identify the context we are associated with
  try {
   cOntext= (Context) event.getLifecycle();
  } catch (ClassCastException e) {
   log.error(sm.getString("contextConfig.cce", event.getLifecycle()), e);
   return;
  }

  // Process the event that has occurred
  if (event.getType().equals(Lifecycle.CONFIGURE_START_EVENT)) {
   configureStart();
  } else if (event.getType().equals(Lifecycle.BEFORE_START_EVENT)) {
   beforeStart();
  } else if (event.getType().equals(Lifecycle.AFTER_START_EVENT)) {
   // Restore docBase for management tools
   if (originalDocBase != null) {
    context.setDocBase(originalDocBase);
   }
  } else if (event.getType().equals(Lifecycle.CONFIGURE_STOP_EVENT)) {
   configureStop();
  } else if (event.getType().equals(Lifecycle.AFTER_INIT_EVENT)) {
   init();
  } else if (event.getType().equals(Lifecycle.AFTER_DESTROY_EVENT)) {
   destroy();
  }
 }

 //......
}

LifecycleSupport是我们需要了解的主要对象,它是监听对象的一个管理类,原理其实和上面的例子差不多,对应了MyEventSource类的部分功能,方便EventSource类来管理监听对象。它把对监听对象的添加移除以及发布事件几个操作进行了统一管理,避免EventSource类中出现太多管理监听对象的逻辑。

public final class LifecycleSupport {

 //......

 //监听对象集合
 private LifecycleListener listeners[] = new LifecycleListener[0];
 
 private final Object listenersLock = new Object(); // Lock object for changes to listeners

 //添加监听对象
 public void addLifecycleListener(LifecycleListener listener) {

  synchronized (listenersLock) {
   LifecycleListener results[] =
   new LifecycleListener[listeners.length + 1];
   for (int i = 0; i 

使用了LifecycleSupport之后,操作LifecycleListener就简单多了,只需要调用LifecycleSupport的各个方法就可以了:

public abstract class LifecycleBase implements Lifecycle{

 //......
 private LifecycleSupport lifecycle = new LifecycleSupport(this);

 @Override
 public void addLifecycleListener(LifecycleListener listener) {
  lifecycle.addLifecycleListener(listener);
 }

 @Override
 public void removeLifecycleListener(LifecycleListener listener) {
  lifecycle.removeLifecycleListener(listener);
 }

 protected void fireLifecycleEvent(String type, Object data) {
  lifecycle.fireLifecycleEvent(type, data);
 }
 //......
}

在需要发布事件时调用fireLifecycleEvent方法就可以发布事件:

fireLifecycleEvent(Lifecycle.CONFIGURE_STOP_EVENT, null);

tomcat事件机制就是在之前的例子上抽出了一个LifecycleSupport类来方便管理监听对象的各种操作,这是一个可以借鉴的地方,其他差别并不大。再来看看spring中对事件机制的处理。

Spring的事件机制

spring中的事件机制原理也是一样的,只是相对来说实现上稍微复杂一点。还是通过相同的角度来看这个问题。

首先是EventObject,spring里面的主要实现是ApplicationEvent:

这里通过ContextStartedEvent类来查看EventObject,它关注的对象是ApplicationContext,是spring容器在启动时触发的事件对象:

public abstract class ApplicationEvent extends EventObject {

 //......
 public ApplicationEvent(Object source) {
  super(source);
  this.timestamp = System.currentTimeMillis();
 }
 //......
}

public abstract class ApplicationContextEvent extends ApplicationEvent {
 public ApplicationContextEvent(ApplicationContext source) {
  super(source);
 }

 public final ApplicationContext getApplicationContext() {
  return (ApplicationContext)this.getSource();
 }
}

public class ContextStartedEvent extends ApplicationContextEvent {
 public ContextStartedEvent(ApplicationContext source) {
  super(source);
 }
}

事件监听接口ApplicationListener,定义了onApplicationEvent方法用来传递监听者感兴趣的ApplicationEvent对象,监听者使用ApplicationEvent参数用来在Context的各个阶段处理进行相应处理。

如果我们需要在容器启动后进行相应处理,那么我们可以在业务类中实现ApplicationListener接口,在事件发生时就会发起通知:

public interface ApplicationListener extends EventListener {

 void onApplicationEvent(E event);
}

public class MyApplicationListener implements ApplicationListener {

 @Override
 public void onApplicationEvent(ApplicationEvent applicationEvent) {

  if (applicationEvent instanceof ContextRefreshedEvent){

   System.out.println("context refresh!"); 
  }
 }
}

那么在spring框架中是怎么发布这些事件的呢?是不是也有一个类似tomcat中LifecycleSupport一样的类呢?通过查看源码可以发现发现,ApplicationContext容器在初始化阶段会调用refresh()方法,这其中又调用了
finishRefresh()方法,这其中调用了publishEvent(new ContextRefreshedEvent(this))方法,发布了ContextRefreshedEvent这一对象。

protected void finishRefresh() {
 
 //......
 // Publish the final event.
 publishEvent(new ContextRefreshedEvent(this));
}

protected void publishEvent(Object event, @Nullable ResolvableType eventType) {

 //......
 getApplicationEventMulticaster().multicastEvent(applicationEvent, eventType);
 //......
}

publishEvent方法通过调用一个默认的多播器SimpleApplicationEventMulticaster的multicastEvent方法来发布各种事件:

SimpleApplicationEventMulticaster
public void multicastEvent(final ApplicationEvent event, @Nullable ResolvableType eventType) {
 ResolvableType type = (eventType != null &#63; eventType : resolveDefaultEventType(event));

 //通过getApplicationListeners获取了所有监听器,然后通过invokeListener方法循环发布事件
 for (final ApplicationListener<&#63;> listener : getApplicationListeners(event, type)) {
  Executor executor = getTaskExecutor();
  if (executor != null) {
   executor.execute(() -> invokeListener(listener, event));
  }
  else {
   invokeListener(listener, event);
  }
 }
}

protected void invokeListener(ApplicationListener<&#63;> listener, ApplicationEvent event) {
 //......
 doInvokeListener(listener, event);
}

private void doInvokeListener(ApplicationListener listener, ApplicationEvent event) {
 //......
 listener.onApplicationEvent(event);
}

也就是说在spring容器中发布ApplicationListener所关注的对象是通过SimpleApplicationEventMulticaster这个类来管理的,和tomcat中LifecycleSupport的功能类似,只是在实现上有略微差别。

最后提一句,在spring中你也可以自己发布各种事件,调用ApplicationContext的publishEvent方法即可。

applicationContext.publishEvent(new ApplicationEvent(new String("事件发布")) { });

总结

这篇文章对Java的事件机制在tomcat以及spring框架中的实现做了一个简单总结和对比,你需要知道以下几点:

  • JDK中定义了EventObject和EventListener两个接口,奠定了事件机制的基础。
  • Tomcat额外提供了一个support类来对监听器的添加删除以及发布进行管理。
  • Spring容器内部通过SimpleApplicationEventMulticaster来发布各个事件,用户可以通过实现ApplicationListener接口来监听自己感兴趣的容器事件。

希望你通过这篇文章的学习可以对Java的事件机制有一个更深刻的认识,在实现自己的事件机制时有可以借鉴以及改进的地方。

好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对的支持。


推荐阅读
  • Eclipse 中 JSP 开发环境配置指南
    本文详细介绍了如何在 Eclipse 集成开发环境中配置 JSP 运行环境,包括必要的软件下载、Tomcat 服务器的配置以及常见问题的解决方法。 ... [详细]
  • 本文介绍了如何利用 Spring Boot 和 Groovy 构建一个灵活且可扩展的动态计算引擎,以满足钱包应用中类似余额宝功能的推广需求。我们将探讨不同的设计方案,并最终选择最适合的技术栈来实现这一目标。 ... [详细]
  • 简化报表生成:EasyReport工具的全面解析
    本文详细介绍了EasyReport,一个易于使用的开源Web报表工具。该工具支持Hadoop、HBase及多种关系型数据库,能够将SQL查询结果转换为HTML表格,并提供Excel导出、图表显示和表头冻结等功能。 ... [详细]
  • 深入解析SpringMVC核心组件:DispatcherServlet的工作原理
    本文详细探讨了SpringMVC的核心组件——DispatcherServlet的运作机制,旨在帮助有一定Java和Spring基础的开发人员理解HTTP请求是如何被映射到Controller并执行的。文章将解答以下问题:1. HTTP请求如何映射到Controller;2. Controller是如何被执行的。 ... [详细]
  • 深入解析Spring启动过程
    本文详细介绍了Spring框架的启动流程,帮助开发者理解其内部机制。通过具体示例和代码片段,解释了Bean定义、工厂类、读取器以及条件评估等关键概念,使读者能够更全面地掌握Spring的初始化过程。 ... [详细]
  • docker镜像重启_docker怎么启动镜像dock ... [详细]
  • 本文探讨了如何通过一系列技术手段提升Spring Boot项目的并发处理能力,解决生产环境中因慢请求导致的系统性能下降问题。 ... [详细]
  • 本文将详细介绍通过CAS(Central Authentication Service)实现单点登录的原理和步骤。CAS由耶鲁大学开发,旨在为多应用系统提供统一的身份认证服务。文中不仅涵盖了CAS的基本架构,还提供了具体的配置实例,帮助读者更好地理解和应用这一技术。 ... [详细]
  • 烤鸭|本文_Spring之Bean的生命周期详解
    烤鸭|本文_Spring之Bean的生命周期详解 ... [详细]
  • 本文详细介绍了如何正确安装Java EE SDK,并解决在安装过程中可能遇到的问题,特别是关于servlet代码在Apache Tomcat 10中无法运行的情况。 ... [详细]
  • 云计算的优势与应用场景
    本文详细探讨了云计算为企业和个人带来的多种优势,包括成本节约、安全性提升、灵活性增强等。同时介绍了云计算的五大核心特点,并结合实际案例进行分析。 ... [详细]
  • 12月16日JavaScript变量、函数、流程、循环等***线上九期班
    12月16日JavaScript变量、函数、流程、循环等***线上九期班 ... [详细]
  • 深入解析Serverless架构模式
    本文将详细介绍Serverless架构模式的核心概念、工作原理及其优势。通过对比传统架构,探讨Serverless如何简化应用开发与运维流程,并介绍当前主流的Serverless平台。 ... [详细]
  • 搭建Jenkins、Ant与TestNG集成环境
    本文详细介绍了如何在Ubuntu 16.04系统上配置Jenkins、Ant和TestNG的集成开发环境,涵盖从安装到配置的具体步骤,并提供了创建Windows Slave节点及项目构建的指南。 ... [详细]
  • 本文详细介绍了 Android 开发中 layout_gravity 属性的使用方法及其在不同布局下的效果,旨在帮助开发者更好地理解和利用这一属性来精确控制视图的布局。 ... [详细]
author-avatar
北达学院树洞
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有