Spring容器的事件监听机制(简单明了的介绍)

本文涉及的产品
容器镜像服务 ACR,镜像仓库100个 不限时长
简介: 上一篇我们介绍了SpringFactoriesLoader。这一篇我接着来介绍一下Spring的另一个知识点,就是Spring容器的事件监听机制。

前言

上一篇我们介绍了SpringFactoriesLoader。这一篇我接着来介绍一下Spring的另一个知识点,就是Spring容器的事件监听机制。

事件

说到事件,我们第一反应是什么是事件?其实 事件是发生在应用程序中的动作,比如点击按钮,在文本框中输入内容等操作都被称为事件。而当事件触发时,应用程序做出的一定的响应则表示应用监听了这个事件,而在服务器端,事件的监听机制更多的用于异步通知以及监控和异常处理。Java提供了实现事件监听机制的两个基础类:自定义事件类型扩展自java.util.EventObject,事件的监听器扩展自java.util.EnventListener。下面我们就以一个监控方法的耗时为例。

1. 定义事件

首先自定义事件类型,通常的做法是继承EnventObject类,随着事件的发生,相应的状态通常封装在此类中。在此处我们定义了一个时间戳,用于记录方法的开始执行时间。

/**
 * 定义事件类型,通常的做法是继承 EnventObject,
 * 随着事件的发生,相应的状态通常都封装在此类中。
 * @author xiang.wei
 * @date 2020/5/5 5:13 PM
 */
public class MethodMonitorEvent extends EventObject {
    /**
     * 时间戳,用于记录方法开始执行的时间
     */
    protected long timestamp;
    /**
     * Constructs a prototypical Event.
     *
     * @param source The object on which the Event initially occurred.
     * @throws IllegalArgumentException if source is null.
     */
    public MethodMonitorEvent(Object source) {
        super(source);
    }
}

2. 定义监听器

事件类型定义好之后,接下来,我们还需要定义一个监听器,用于监听事件的发生。我们可以在方法开始执行之前发布一个begin事件,在方法结束之后发布一个end事件。我们定义了事件监听接口,里面定义了处理begin事件的方法onMethodBegin和处理end事件的方法onMethodEnd。我们注意到这两个方法都只有一个参数,就是MethodMonitorEvent 参数。说明这个监听器类只负责监听对应的事件并进行处理。

/**
 * 定义事件监听接口
 * @author xiang.wei
 * @date 2020/5/5 5:24 PM
 */
public interface MethodMonitorEventListener extends EventListener {
    /**
     * 处理方法执行之前发布的事件
     * @param event
     */
    void onMethodBegin(MethodMonitorEvent event);
    /**
     * 处理方法结束时发布的事件
     * @param event
     */
    void onMethodEnd(MethodMonitorEvent event);
}

在这个监听器接口的实现类里我们将会具体实现这两个方法的逻辑。方法也是很简单

public class MethodMonitorEventListenerImpl implements MethodMonitorEventListener {
    @Override
    public void onMethodBegin(MethodMonitorEvent event) {
        //记录方法开始执行时的时间
        event.timestamp = System.currentTimeMillis();
    }
    @Override
    public void onMethodEnd(MethodMonitorEvent event) {
        //计算耗时
        long duration = System.currentTimeMillis() - event.timestamp;
        System.out.println("总耗时:" + duration+" ms");
    }
}

3. 定义发布器

有了事件和监听器。我们还需要一个事件发布者,它本身作为一个事件源,在合适的时机,将相应的时间发布给对应的事件监听器。

public class MethodMonitorEventPublisher {
    private List<MethodMonitorEventListener> eventListeners = new ArrayList<>();
    public void methodMonitor() throws InterruptedException {
        //定义事件
        MethodMonitorEvent eventObject = new MethodMonitorEvent(this);
        publishEvent("begin", eventObject);
        //模拟方法执行:休眠5秒钟
        TimeUnit.SECONDS.sleep(5);
        publishEvent("end", eventObject);
    }
  //发布事件的逻辑
    public void publishEvent(String status, MethodMonitorEvent event) {
        //避免在事件处理期间,监听器被移除,这里为了安全做了一个复制操作
        List<MethodMonitorEventListener> copyListeners = new ArrayList<>(eventListeners);
        for (MethodMonitorEventListener listener : copyListeners) {
            if ("begin".equals(status)) {
                listener.onMethodBegin(event);
            } else {
                listener.onMethodEnd(event);
            }
        }
    }
    public static void main(String[] args) throws InterruptedException {
        MethodMonitorEventPublisher publisher = new MethodMonitorEventPublisher();
        //添加监听器
        publisher.addEventListener(new MethodMonitorEventListenerImpl());
  //发布事件
        publisher.methodMonitor();
    }
    public void addEventListener(MethodMonitorEventListener listener) {
        eventListeners.add(listener);
    }
}

对于事件发布者(事件源)我们需要关注两点:

在合适的时机发布事件,此例中的methodMonitor()方法就是事件发布的源头,其在方法执行之前和结束之后两个时间点发布MethodMonitorEvent事件,每个时间点发布的事件都会传给相应的监听器进⾏处理。

事件监听器的管理:publisher 类中提供了事件监听器的注册和移除方法。这样客户端可以根据实际情况决定是否需要注册新的监听器或者移除某个监听器,如果没有提供remove方法,那么注册的监听器实例将一直被MethodMonitorEventPublisher引⽤,即使已经废弃不⽤了,也依然在发布者的监听器列表中,这会导致隐性的内存泄漏。

Spring容器的事件监听机制

说完了Java提供的事件监听机制的两个基础类,以及如何实现一个自定义事件的demo。下面就请出本篇文章的主角Spring容器的时间监听机制。

Spring的ApplicationContext容器内部中的所有事件类型均继承自org.springframework.context.ApplicationEvent

容器中的所有监听器都实现了org.springframework.context.ApplicationListener接口

并且以bean的形式注册到了容器中,一旦容器内发布ApplicationEvent及其子类型的事件,注册到容器中的ApplicationListener就会对这些事件进行处理。ApplicationEventMulticaster类是事件管理者,管理监听器和发布事件,ApplicationContext通过委托ApplicationEventMulticaster来发布事件

ApplicationEventPublisher 是事件发布者,该接口封装了事件有关的公共方法。

1.事件的继承类图

ApplicationContextEvent 继承自ApplicationEvent,而ApplicationEvent 继承自EventObject,Spring提供了一些默认的实现,比如:

ContextStartedEvent 表示容器在启动时发布的事件类型,即调用start()方法。

ContextRefreshedEvent表示容器在初始化或者刷新的时候发布的事件类型,如调用refresh() 方法,此处的实例化是指所有的bean都已经被加载。后置处理器被激活。所有单例bean都已被实例化。所有的容器对象都已准备好可使用。

ContextStoppedEvent表示容器在即将关闭时发布的事件类型,即调用了stop()方法。

监听器的继承类图

容器内部用ApplicationListener作为事件监听器接口定义,它继承自EventListener。ApplicationContext容器在启动时,会自动识别并加载EventListener类型的bean,一旦容器内有事件发布,将通知这些注册到容器的EventListener。

其中ContextRefreshListener监听器监听的是ContextRefreshedEvent事件,而ContextCloserListener监听器监听的是ContextClosedEvent事件。

private class ContextRefreshListener implements ApplicationListener<ContextRefreshedEvent> {
  @Override
  public void onApplicationEvent(ContextRefreshedEvent event) {
    FrameworkServlet.this.onApplicationEvent(event);
  }
  }

而ApplicationContext接口继承了ApplicationEventPublisher接口,该接口提供了void publishEvent(ApplicationEvent event)方法的定义,不难看出,ApplicationContext容器担当的就是事件发布者的角色。需要说明的是Spring事件默认是同步的,即调用publishEvent方法发布事件后,它会处于阻塞状态,直到onApplicationEvent接受到事件并处理返回之后才继续执行下去,这种单线程同步的好处是可以进行任务管理。

总结

本文首先介绍了Java中事件监听机制的基本概念,并且以一个记录方法耗时的demo说明了如何自定义事件类型。接着就是介绍了Spring容器的事件监听机制。


相关文章
|
17天前
|
XML Java 数据格式
Spring容器Bean之XML配置方式
通过对以上内容的掌握,开发人员可以灵活地使用Spring的XML配置方式来管理应用程序的Bean,提高代码的模块化和可维护性。
54 6
|
5月前
|
XML Java 数据格式
Spring5入门到实战------7、IOC容器-Bean管理XML方式(外部属性文件)
这篇文章是Spring5框架的实战教程,主要介绍了如何在Spring的IOC容器中通过XML配置方式使用外部属性文件来管理Bean,特别是数据库连接池的配置。文章详细讲解了创建属性文件、引入属性文件到Spring配置、以及如何使用属性占位符来引用属性文件中的值。
Spring5入门到实战------7、IOC容器-Bean管理XML方式(外部属性文件)
|
1月前
|
安全 Java 开发者
Spring容器中的bean是线程安全的吗?
Spring容器中的bean默认为单例模式,多线程环境下若操作共享成员变量,易引发线程安全问题。Spring未对单例bean做线程安全处理,需开发者自行解决。通常,Spring bean(如Controller、Service、Dao)无状态变化,故多为线程安全。若涉及线程安全问题,可通过编码或设置bean作用域为prototype解决。
34 1
|
2月前
|
前端开发 Java Docker
使用Docker容器化部署Spring Boot应用程序
使用Docker容器化部署Spring Boot应用程序
|
2月前
|
Java Docker 微服务
利用Docker容器化部署Spring Boot应用
利用Docker容器化部署Spring Boot应用
55 0
|
3月前
|
Java 测试技术 Windows
咦!Spring容器里为什么没有我需要的Bean?
【10月更文挑战第11天】项目经理给小菜分配了一个紧急需求,小菜迅速搭建了一个SpringBoot项目并完成了开发。然而,启动测试时发现接口404,原因是控制器包不在默认扫描路径下。通过配置`@ComponentScan`的`basePackages`字段,解决了问题。总结:`@SpringBootApplication`默认只扫描当前包下的组件,需要扫描其他包时需配置`@ComponentScan`。
|
4月前
|
XML Java 开发者
经典面试---spring IOC容器的核心实现原理
作为一名拥有十年研发经验的工程师,对Spring框架尤其是其IOC(Inversion of Control,控制反转)容器的核心实现原理有着深入的理解。
169 3
|
3月前
|
XML Java 数据格式
Spring IOC容器的深度解析及实战应用
【10月更文挑战第14天】在软件工程中,随着系统规模的扩大,对象间的依赖关系变得越来越复杂,这导致了系统的高耦合度,增加了开发和维护的难度。为解决这一问题,Michael Mattson在1996年提出了IOC(Inversion of Control,控制反转)理论,旨在降低对象间的耦合度,提高系统的灵活性和可维护性。Spring框架正是基于这一理论,通过IOC容器实现了对象间的依赖注入和生命周期管理。
86 0
|
5月前
|
XML Java 数据格式
Spring5入门到实战------3、IOC容器-Bean管理XML方式(一)
这篇文章详细介绍了Spring框架中IOC容器的Bean管理,特别是基于XML配置方式的实现。文章涵盖了Bean的定义、属性注入、使用set方法和构造函数注入,以及如何注入不同类型的属性,包括null值、特殊字符和外部bean。此外,还探讨了内部bean的概念及其与外部bean的比较,并提供了相应的示例代码和测试结果。
Spring5入门到实战------3、IOC容器-Bean管理XML方式(一)
|
5月前
|
XML Java 数据格式
Spring5入门到实战------5、IOC容器-Bean管理(三)
这篇文章深入探讨了Spring5框架中IOC容器的高级Bean管理,包括FactoryBean的使用、Bean作用域的设置、Bean生命周期的详细解释以及Bean后置处理器的实现和应用。
Spring5入门到实战------5、IOC容器-Bean管理(三)