【Spring Boot系列】- Spring Boot侦听器Listener

【Spring Boot系列】- Spring Boot侦听器Listener

文章目录

  • [【Spring Boot系列】- Spring Boot侦听器Listener](#【Spring Boot系列】- Spring Boot侦听器Listener)
    • 一、概述
    • 二、监听器Listener分类
      • [2.1 监听ServletContext的事件监听器](#2.1 监听ServletContext的事件监听器)
      • [2.2 监听HttpSeesion的事件监听器](#2.2 监听HttpSeesion的事件监听器)
      • [2.3 监听ServletRequest的事件监听器](#2.3 监听ServletRequest的事件监听器)
    • 三、SpringMVC中的监听器
      • [3.1 ContextLoaderListener](#3.1 ContextLoaderListener)
      • [3.2 RequestContextListen](#3.2 RequestContextListen)
      • [3.3 IntrospectorCleanupListener](#3.3 IntrospectorCleanupListener)
      • [四、Servlet API中Web事件监听器](#四、Servlet API中Web事件监听器)
      • [4.1 ServletContextListener -- 监听servletContext对象的创建以及销毁](#4.1 ServletContextListener -- 监听servletContext对象的创建以及销毁)
      • [4.2 HttpSessionListener -- 监听session对象的创建以及销毁](#4.2 HttpSessionListener -- 监听session对象的创建以及销毁)
      • [4.3 ServletRequestListener -- 监听request对象的创建以及销毁](#4.3 ServletRequestListener -- 监听request对象的创建以及销毁)
      • [4.4 ServletContextAttributeListener -- 监听servletContext对象中属性的改变](#4.4 ServletContextAttributeListener -- 监听servletContext对象中属性的改变)
      • [4.5 HttpSessionAttributeListener --监听session对象中属性的改变](#4.5 HttpSessionAttributeListener --监听session对象中属性的改变)
      • [4.6 ServletRequestAttributeListener --监听request对象中属性的改变](#4.6 ServletRequestAttributeListener --监听request对象中属性的改变)

一、概述

什么事Web监听器?web监听器就是Servlet中特殊的类,他们能帮助开发者监听web中的特定事件,比如ServletContext、HttpSession、ServletRequest的创建和销毁;变量的创建、销毁和修改等。可以在某些动作前后增加处理,实现监控等等。web监听器的使用场景有很多;Spring的监听器是一种观察者模式,它能实现事件与事件监听者直接的解耦;

二、监听器Listener分类

2.1 监听ServletContext的事件监听器

分别为:ServletContextListener、ServletContextAttributeListener。Application级别,整个应用只存在一个,可以进行全局配置。

2.2 监听HttpSeesion的事件监听器

分别为:HttpSessionListener、HttpSessionAttributeListener。Session级别,针对每一个对象,如统计会话总数。

2.3 监听ServletRequest的事件监听器

分别为:ServletRequestListener、ServletRequestAttributeListener。Request级别,针对每一个客户请求。

三、SpringMVC中的监听器

3.1 ContextLoaderListener

在启动Web容器时,自动装配Spring applicationContext.xml的配置信息。因为它实现了ServletContextListener这个接口,在web.xml配置这个监听器,启动容器时,就会默认执行它实现的方法。在ContextLoaderListener中关联了ContextLoader这个类,所以整个加载配置过程由ContextLoader来完成。

以下是web.xml中ContextLoaderListener的配置和context的配置

xml 复制代码
<listener>
   <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>

context-param

xml 复制代码
<context-param>
    <param-name>contextConfigLocation</param-name>
    <param-value>classpath:applicationContext.xml</param-value>
</context-param>

ServletContextListener 是 ServletContext 的监听者,如果 ServletContext 发生变化,如服务器启动时ServletContext 被创建,服务器关闭时 ServletContext 将要被销毁。

3.2 RequestContextListen

基于LocalThread将HTTP request对象绑定到为该请求提供服务的线程上。这使得具有request和session作用域的bean能够在后面的调用链中被访问到。

Request作用域

xml 复制代码
<bean id="loginAction" class="com.goyeer.LoginAction" scope="request"/>  

针对每次HTTP请求,Spring容器会根据loginAction bean定义创建一个全新的LoginAction bean实例,且该loginAction bean实例仅在当前HTTP request内有效,因此可以根据需要放心的更改所建实例的内部状态,而其他请求中根据loginAction bean定义创建的实例,将不会看到这些特定于某个请求的状态变化。当处理请求结束,request作用域的bean实例将被销毁。

Session作用域

xml 复制代码
<bean id="user" class="com.goyeer.User" scope="session"/>

针对某个HTTP Session,Spring容器会根据userPreferences bean定义创建一个全新的userPreferences bean实例,且该userPreferences bean仅在当前HTTP Session内有效。与request作用域一样,你可以根据需要放心的更改所创建实例的内部状态,而别的HTTP Session中根据userPreferences创建的实例,将不会看到这些特定于某个HTTP Session的状态变化。当HTTP Session最终被废弃的时候,在该HTTP Session作用域内的bean也会被废弃掉。

global session作用域

xml 复制代码
<bean id="userPrefere" class="com.goyeer.userPrefere" scope="globalSession"/> 

global session作用域类似于标准的HTTP Session作用域,不过它仅仅在基于portlet的web应用中才有意义。Portlet规范定义了全局Session的概念,它被所有构成某个portlet web应用的各种不同的portlet所共享。在global session作用域中定义的bean被限定于全局portlet Session的生命周期范围内。 请注意,假如你在编写一个标准的基于Servlet的web应用,并且定义了一个或多个具有global session作用域的bean,系统会使用标准的HTTP Session作用域,并且不会引起任何错误。

3.3 IntrospectorCleanupListener

Introspector作用及影响

在分析IntrospectorCleanupListener之前,先了解一下Introspector。Introspector是JDK中java.beans包下的类,它为目标JavaBean提供了一种了解原类方法、属性和事件的标准方法。通俗的说,就是可以通过Introspector构建一个BeanInfo对象,而这个BeanInfo对象中包含了目标类中的属性、方法和事件的描述信息,然后可以使用这个BeanInfo对象对目标对象进行相关操作。

Introspector间接持有了BeanInfo的强引用。如果使用Introspector操作了很多类,那么Introspector将间接持有这些BeanInfo的强引用。在发生垃圾收集的时候,检测到这些BeanInfo存在引用链,则这些类和对应的类加载器将不会被垃圾收集器回收,进而导致内存泄漏。所以,为了解决这个问题,在使用Introspector操作完成后,调用Introspector类的flushCaches方法清除缓存。

配置IntrospectorCleanupListener

此监听器主要用于解决java.beans.Introspector导致的内存泄漏的问题。JDK中的java.beans.Introspector类的用途是发现Java类是否符合JavaBean规范。如果有的框架或程序用到了Introspector类,那么就会启用一个系统级别的缓存,此缓存会存放一些曾加载并分析过的JavaBean的引用。当Web服务器关闭时,由于此缓存中存放着这些JavaBean的引用,所以垃圾回收器无法回收Web容器中的JavaBean对象,最后导致内存变大。而org.springframework.web.util.IntrospectorCleanupListener就是专门用来处理Introspector内存泄漏问题的辅助类。IntrospectorCleanupListener会在Web服务器停止时清理Introspector缓存,使那些Javabean能被垃圾回收器正确回收。Spring自身不会出现这种问题,因为Spring在加载并分析完一个类之后会马上刷新。java.beans.-Introspector缓存,这就保证Spring中不会出现这种内存泄漏的问题。但有些程序和框架在使用了JavaBeans Introspector之后,没有进行清理工作(如Quartz,Struts),最后导致内存泄漏。

​ 在以往的工作经历中,多次看到在web.xml中将IntrospectorCleanupListener配置成非第一个listener。官方的表述是必须将此监听器配置成web.xml中的第一个listener,才能在合适的时间发挥最有效的作用。原因其实很简单,在Servlet3.0规范之前,监听器的调用是随机的,而从Servlet3.0开始,监听器的调用顺序是根据其在web.xml中配置的顺序,并且实现ServletContextListener的监听器,contextInitialized方法调用顺序是按照在web.xml中配置的顺序正序依次执行,而contextDestroyed方法的调用顺序是按照在web.xml中配置的顺序逆序依次执行。所以,如果Introspector-CleanupListener被配置成了第一个listener,那么它的contextDestroyed方法将最后一个执行,将发挥最有效的清除作用;而如果不是,那么可能会残留未被清除的缓存。

四、Servlet API中Web事件监听器

4.1 ServletContextListener -- 监听servletContext对象的创建以及销毁

  • contextInitialized(ServletContextEvent arg0) -- 创建时执行
  • contextDestroyed(ServletContextEvent arg0) -- 销毁时执行

4.2 HttpSessionListener -- 监听session对象的创建以及销毁

  • sessionCreated(HttpSessionEvent se) -- 创建时执行
  • sessionDestroyed(HttpSessionEvent se) -- 销毁时执行

4.3 ServletRequestListener -- 监听request对象的创建以及销毁

  • requestInitialized(ServletRequestEvent sre) -- 创建时执行
  • requestDestroyed(ServletRequestEvent sre) -- 销毁时执行

4.4 ServletContextAttributeListener -- 监听servletContext对象中属性的改变

  • attributeAdded(ServletContextAttributeEvent event) -- 添加属性时执行
  • attributeReplaced(ServletContextAttributeEvent event) -- 修改属性时执行
  • attributeRemoved(ServletContextAttributeEvent event) -- 删除属性时执行

4.5 HttpSessionAttributeListener --监听session对象中属性的改变

  • attributeAdded(HttpSessionBindingEvent event) -- 添加属性时执行
  • attributeReplaced(HttpSessionBindingEvent event) -- 修改属性时执行
  • attributeRemoved(HttpSessionBindingEvent event) -- 删除属性时执行

4.6 ServletRequestAttributeListener --监听request对象中属性的改变

  • attributeAdded(ServletRequestAttributeEvent srae) -- 添加属性时执行
  • attributeReplaced(ServletRequestAttributeEvent srae) -- 修改属性时执行
  • attributeRemoved(ServletRequestAttributeEvent srae) -- 删除属性时执行
相关推荐
ok!ko3 小时前
设计模式之原型模式(通俗易懂--代码辅助理解【Java版】)
java·设计模式·原型模式
2401_857622663 小时前
SpringBoot框架下校园资料库的构建与优化
spring boot·后端·php
2402_857589364 小时前
“衣依”服装销售平台:Spring Boot框架的设计与实现
java·spring boot·后端
吾爱星辰4 小时前
Kotlin 处理字符串和正则表达式(二十一)
java·开发语言·jvm·正则表达式·kotlin
哎呦没5 小时前
大学生就业招聘:Spring Boot系统的架构分析
java·spring boot·后端
编程、小哥哥5 小时前
netty之Netty与SpringBoot整合
java·spring boot·spring
IT学长编程6 小时前
计算机毕业设计 玩具租赁系统的设计与实现 Java实战项目 附源码+文档+视频讲解
java·spring boot·毕业设计·课程设计·毕业论文·计算机毕业设计选题·玩具租赁系统
莹雨潇潇6 小时前
Docker 快速入门(Ubuntu版)
java·前端·docker·容器
杨哥带你写代码6 小时前
足球青训俱乐部管理:Spring Boot技术驱动
java·spring boot·后端
郭二哈7 小时前
C++——模板进阶、继承
java·服务器·c++