03启动spring

简介: 1.配置Servlet初始化参数contextConfigLocation,FrameworkServlet获取设置的参数值2.启动spring生命周期,并注册回调监听

内容概览


  • 配置Servlet初始化参数contextConfigLocation,FrameworkServlet获取设置的参数值
  • 启动spring生命周期,并注册回调监听


为Servlet设置初始化参数,同时FrameworkServlet获取初始化的参数值


到目前为止上篇创建的DispatcherServlet类和spring还没半毛钱的关系,最多就是包名和类名相同了。既然要使用spring那么spring的配置文件必然少不了且该配置文件是通过Servlet的初始化参数设置的。


<?xml version="1.0" encoding="UTF-8"?>

<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"

        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

        xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"

        version="4.0">



   <!--    设置Springmvc前端处理器-->

   <servlet>

       <servlet-name>DispatcherServlet</servlet-name>

       <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>

       <!--设置Servlet初始化参数-->

       <init-param>

           <param-name>contextConfigLocation</param-name>

           <param-value>classpath:spring-mvc-config.xml</param-value>

       </init-param>

       <!--指定Servlet启动顺序-->

       <load-on-startup>1</load-on-startup>

   </servlet>


   <servlet-mapping>

       <servlet-name>DispatcherServlet</servlet-name>

       <url-pattern>/*</url-pattern>

   </servlet-mapping>


</web-app>


还是在web.xml中通过init-param为Servlet设置初始化参数。获取参数时会用到下列方法


  • getInitParameter(String name)    //获取初始化参数值
  • getInitParameterNames()    //以String对象的Enumeration的形式返回 servlet的初始化参数的名称
  • getServletName()    //返回此 servlet 实例的名称


初始化参数contextConfigLocation在spring体系里是由FrameworkServlet维护的。设置流程如下图:



  1. HttpServletBean重写了Servlet的init方法,所以会在tomcat启动的生命周期里执行
  2. 在init方法中将获取到的初始化参数调用spring-beans模块中的属性访问器功能模块去设置到对应的位置
  1. BeanWrapper相当于一个代理器,提供了为JavaBean设置获取属性的功能
  1. 初始化参数contextConfigLocation的set方法在FrameworkServlet类中维护


debug查看其调用栈如下图:



代码实现


HttpServletBean的init方法


   @Override

   public void init() throws ServletException {

       //1.获取Servlet设置的初始化参数

       PropertyValues pvs = new ServletConfigPropertyValues(getServletConfig(), this.requiredProperties);

       if (!pvs.isEmpty()) {


           try {

               BeanWrapper bw = PropertyAccessorFactory.forBeanPropertyAccess(this);

               //2.这里将获取到的参数交给bw去设置到对应的set方法

               bw.setPropertyValues(pvs, true);

           } catch (BeansException e) {

               if (logger.isErrorEnabled()) {

                   logger.error("Failed to set bean properties on servlet '" + getServletName() + "'", e);

               }

               throw e;

           }

       }


       //3.在初始化参数处理完成后,调用模版方法让子类进行初始化

       initServletBean();

   }


   /**

    * Subclasses may override this to perform initialization. All bean

    * properties of this servlet will have been set before this method is invoked.

    */

   protected void initServletBean() throws ServletException {


   }


  1. ServletConfigPropertyValues是个内部类,去获取为Servlet配置的初始化参数,同时校验是否设置了必要的参数。requiredProperties是一个Set集合可以设置约束哪些参数是必要的,如果没有设置则Servlet启动失败
  2. 将获取到的参数键值对交给BeanWrapper为处理。这里省略了部分暂时不用到的代码。
  3. 在初始化参数设置完成后,调用定义的模版方法initServletBean进行下一步的操作


FrameworkServlet的setContextConfigLocation方法


在web.xml中设置的初始化参数key为contextConfigLocation,所以就必须有对应的get、set方法


  • getContextConfigLocation
  • setContextConfigLocation


   @Nullable

   private String contextConfigLocation;



   @Nullable

   public String getContextConfigLocation() {

       return contextConfigLocation;

   }


   public void setContextConfigLocation(@Nullable String contextConfigLocation) {

       this.contextConfigLocation = contextConfigLocation;

   }


测试


debug启动example-easy-spring可以看到setContextConfigLocation方法被调用



启动spring生命周期,并注册回调监听


既然已经为spring设置了配置文件,那么必然得有调用的地方不然干嘛费尽周折进行设置。理所应当也就是在启动spring时调用getContextConfigLocation方法。


启动spring流程图如下所示:



  1. HttpServletBean重写了Servlet的init方法,所以会按照Servlet的生命周期执行init方法。在init方法里处理完Servlet的初始化参数之后,会调用模版方法initServletBean
  2. FrameworkServlet重写了initServletBean方法,经过层层处理在该类的configureAndRefreshWebapplicationContext方法里触发了spring的启动


代码实现


FrameworkServlet中的系列方法


protected WebApplicationContext createWebApplicationContext(@Nullable ApplicationContext parent) {

       //获取上下文的class对象,即XmlWebApplicationContext.class

       Class<?> contextClass = getContextClass();

       if (!ConfigurableWebApplicationContext.class.isAssignableFrom(contextClass)) {

           throw new ApplicationContextException(

                   "Fatal initialization error in servlet with name '" + getServletName() +

                           "': custom WebApplicationContext class [" + contextClass.getName() +

                           "] is not of type ConfigurableWebApplicationContext"

           );

       }


       //构造一个XmlWebApplicationContext对象

       ConfigurableWebApplicationContext wac = (ConfigurableWebApplicationContext) BeanUtils.instantiateClass(contextClass);


       wac.setParent(parent);


       //获取设置的spring配置文件

       String configLocation = getContextConfigLocation();

       if (null != configLocation) {

           wac.setConfigLocation(configLocation);

       }

       configureAndRefreshWebApplicationContext(wac);


       return wac;

   }


   protected void configureAndRefreshWebApplicationContext(ConfigurableWebApplicationContext wac) {


       //注册回调监听

       wac.addApplicationListener(new SourceFilteringListener(wac, new ContextRefreshListener()));


       wac.refresh();

   }


   /**

    * ApplicationListener endpoint that receives events from this servlet's

    * WebApplicationContext only, delegating to onApplicationEvent on the

    * FrameworkServlet instance.

    */

   private class ContextRefreshListener implements ApplicationListener<ContextRefreshedEvent> {


       @Override

       public void onApplicationEvent(ContextRefreshedEvent event) {

           logger.info("接收回调事件");

       }

   }


默认情况下spring配置文件是交给web模块中的XmlWebApplicationContext来进行解析的。


其中refresh方法是启动spring的入口,在启动spring之前首先设置了spring配置文件然后注册了一个上下文监听事件。当spring启动之后会回调内部类实现的回调监听端点做后续的初始化操作。


XmlWebApplicationContext的类结构


refresh方法真正的实现是在AbstractApplicationContext类中,这块逻辑在单独描述吧不属于当前的范围内。XmlWebApplicationContext算是应用上下文在web层面使用xml处理的一个类,是AbstractApplicationContext的子类所以可以调用refresh方法。



这里之所以把这个类关系图列出来主要是描述一点,在spring实现过程中有很多这种模式。一个类A实现了接口B,但是接口B中定义的方法却是让类A的父类C实现的。稍微有点绕抽象成简单的类关系之后代码如下:


public interface Function {

   void speak();

}


public class Parent {


   public void speak(){


   }

}


public class Child extends Parent implements Function {



}


测试


debug启动example-easy-spring项目效果如下:


相关文章
|
IDE 网络协议 Java
2021最新 IDEA 启动失败 & 启动Spring boot 项目端口被占用问题 彻底解决方案
2021最新 IDEA 启动失败 & 启动Spring boot 项目端口被占用问题 彻底解决方案
819 0
2021最新 IDEA 启动失败 & 启动Spring boot 项目端口被占用问题 彻底解决方案
|
JSON SpringCloudAlibaba 负载均衡
【微服务35】分布式事务Seata源码解析三:从Spring Boot特性来看Seata Client 启动时都做了什么
【微服务35】分布式事务Seata源码解析三:从Spring Boot特性来看Seata Client 启动时都做了什么
854 0
【微服务35】分布式事务Seata源码解析三:从Spring Boot特性来看Seata Client 启动时都做了什么
|
IDE Java 测试技术
Spring 5 启动性能优化之 @Indexed
背景 Spring 经过近20年的发展,目前版本已经迭代到了5.x,每个版本 Spring 都有不同的改进,版本 5.x 中,Spring 把重心放到了性能优化上。我们知道,Spring 注解驱动编程中,Spring 启动时需要对类路径下的包进行扫描,以便发现所需管理的 bean。如果在应用启动前能够确定 Spring bean,不再进行扫描,那么性能就会大大提高,Spring 5 对此进行了实现。
1179 0
Spring 5 启动性能优化之 @Indexed
|
Java Maven Spring
Elastic实战:项目中已经剔除了spring data elasticsearch依赖,但启动项目仍然会进行es健康检查
在实际开发中遇到一个问题:原本在springboot项目中引入了spring data elasticsearch的依赖,后因调整将这个依赖从这个服务中删除了,但是启动服务仍然会进行es的健康检查。也就导致一直有警告日志输出:connection refuse
203 0
Elastic实战:项目中已经剔除了spring data elasticsearch依赖,但启动项目仍然会进行es健康检查
|
Java Spring 容器
Spring Boot 启动时自动执行代码的几种方式。。
Spring Boot 启动时自动执行代码的几种方式。。
534 0
Spring Boot 启动时自动执行代码的几种方式。。
|
安全 IDE Java
一张图帮你记忆,Spring Boot 应用在启动阶段执行代码的几种方式
一张图帮你记忆,Spring Boot 应用在启动阶段执行代码的几种方式
一张图帮你记忆,Spring Boot 应用在启动阶段执行代码的几种方式
|
Java 索引 Spring
spring data elasticsearch:启动项目时自动创建索引
在springboot整合spring data elasticsearch项目中,当索引数量较多,mapping结构较为复杂时,我们常常希望启动项目时能够自动创建索引及mapping,这样就不用再到各个环境中创建索引了 所以今天咱们就来看看如何自动创建索引
1523 0
|
Java Spring 容器
简析SpringBoot启动执行流程
简析一下SpringBoot的启动流程
简析SpringBoot启动执行流程
|
负载均衡 Java 微服务
Spring Cloud Ribbon 全解 (6) - SpringCloud环境下纯Ribbon(不包含Eureka)使用与启动分析
Spring Cloud Ribbon 全解 (6) - SpringCloud环境下纯Ribbon(不包含Eureka)使用与启动分析
Spring Cloud Ribbon 全解 (6) - SpringCloud环境下纯Ribbon(不包含Eureka)使用与启动分析