DispatchServlet初始化源码分析

写在前面

本文分为两大板块

  • 监听器ContextLoaderListener源码分析

  • DispatchServlet初始化源码分析

容器启动时执行的顺序

web.xml中定义的绝大多数东西是随着容器的启动而执行的,比如servlet,filter,listener,contextParam,具体的执行顺序为 contextParam->listener->filter->servlet

监听器ContextLoaderListener源码分析

我们在写SpringMVC项目时都需要在web.xml配置一个listener,我们就从这个listenser开始,看看内部究竟发生了什么。


     org.springframework.web.context.ContextLoaderListener
 

如下为我们配置的监听器ContextLoaderListener的继承关系图。

ContextLoaderListener继承关系图

可以看到ContextLoaderListener继承了ContextLoader并实现了ServletContextListener接口。

每个实现ServletContextListener的监听器都必须实现如下两个方法(contextInitialized()和contextDestroyed()),作用我们从名字上就可以看出来,分别是容器启动时做一些初始化工作和容器关闭时做一些清理工作。

如下为ContextLoaderListener.java代码。

public class ContextLoaderListener extends ContextLoader implements ServletContextListener {
  public void contextInitialized(ServletContextEvent event) {
    //初始化Root WebApplicationContext
        initWebApplicationContext(event.getServletContext());
    }
  public void contextDestroyed(ServletContextEvent event) {
        closeWebApplicationContext(event.getServletContext());
        ContextCleanupListener.cleanupAttributes(event.getServletContext());
    }
}

这里initWebApplicationContext()方法调用的是父类的ContextLoad.initWebApplicationContext()

//ContextLoad.initWebApplicationContext()
public WebApplicationContext initWebApplicationContext(ServletContext servletContext) {
  ...
  try {
    if (this.context == null) {
      //this.conext即为Root WebApplicationContext
      //如果没有配置WebApplicationContext的实现类,将使用默认的XmlWebApplicationContext实现类创建WebApplicationContext对象
      //传入servletContext目的是为了读取配置文件中的context_class参数
      this.context = createWebApplicationContext(servletContext);
    }
    if (this.context instanceof ConfigurableWebApplicationContext) {
      ConfigurableWebApplicationContext cwac = (ConfigurableWebApplicationContext) this.context;
      if (!cwac.isActive()) {
        if (cwac.getParent() == null) {
          //设置Root WebApplicationContext的parent,作用个人猜想可能是跟分布式应用有关,
          //分布式应用每个分应用都有一个Root WebApplicationContext,现在如果这么多Root WebApplicationContext
          //需要共享数据的话就需要一个共同的parent来保存共享的数据了
          //在单应用中parent为null
          ApplicationContext parent = loadParentContext(servletContext);
          cwac.setParent(parent);
        }
        //将Root WebApplicationContext与ServletContext建立关联,读取applicationContext.xml文件并配置Root WebApplicationContext
        configureAndRefreshWebApplicationContext(cwac, servletContext);
      }
    }
    //ServletContext与Root WebApplicationContext建立关联
    servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, this.context);
    ...
    return this.context;
  }
}

代码中出现了如下对象 WebApplicationContext,ConfigurableWebApplicationContext , ApplicationContext以及 ServletContext。

  1. 先来说前二者WebApplicationContext,ConfigurableWebApplicationContext之间的关系:

ConfigurableWebApplicationContext扩展了WebApplicationContext,它允许通过配置化的方式实例化WebApplicationContext。同时定义了两个重要的方法。

  • setServletContext(): 为Spring设置Web应用的上下文,以便二者整合。

  • setConfigLocations(): 设置Spring配置文件地址。

  1. 接着是ServletContext 和 WebApplicationContext之间的关系。
ServletContextAndWebApplicationContext
  1. 最后为了说明上述的Root WebApplicationContext 和 ApplicationContext parent的关系,我在Spring官网上找到了这么一张图。
上下文关系

简单说明各个 WebApplicationContext 的功能:

  • WebApplicationContext: 与 dispatchServlet 直接相关,通过xxx-servlet.xml文件配置,是 dispatchServlet 的上下文,包含了各种控制器(Controllers),视图解析器(ViewResolver)以及映射器(HandlerMapping)。

  • Root WebApplicationContext: 单应用下为一个,分布式应用会存在多个。通过applicationContext.xml配置。包含各种业务逻辑以及对数据库进行的操作。

  • parent: 分布式应用中才会有,为多个共享 Root WebApplicationContext 而生。

至此,我们监听器ContextLoaderListener的工作就完成了,我们总结如下:

  1. 创建Root WebApplicationContext并通过ServletContext完成配置。
  2. 如果是分布式应用将Root WebApplicationContext与parent建立关系。
  3. 完成ServletContext与Root WebApplicationContext之间的相互关联。

DispatchServlet的初始化

有关 dispatchServlet 的继承关系如图所示:

dispatchServlet继承关系

可以看到,HttpServletBean 和 FramworkServlet 是 dispatchServlet的父类,并且他们都是 HttpServlet的子类。
要想初始化 DispatchServlet,必须先创建出一系列父类对象。

在一个servlet能接受请求并发出响应之前,它需要先完成初始化工作(调用init()方法)。我们在web.xml中只配置了一个servlet(即 dispatchServlet),它随着容器的启动而启动,我们再次强调前面提到过执行顺序。
contextParam->listener->filter->servlet

可以看到,servlet在listener之后执行,所以在调用 servlet.init() 方法之前,Root WebApplicationContext已经完成初始化,而 dispatchServlet 初始化的工作就是 完成 WebApplicationContext的初始化。

dispatchServlet 中的init()方法,继承自父类 HttpSrevletBean 中定义的 init()方法。

如下是 HttpSrevletBean 中的 init()方法,整个 DispatchServlet 的初始化也由此开始。

  1. 继承自HttpServletBean的init()方法
public final void init() throws ServletException {
  ...
  try {
    //读取xxx-servlet.xml
      PropertyValues pvs = new ServletConfigPropertyValues(getServletConfig(), this.requiredProperties);
    //创建BeanWrapper对象
      BeanWrapper bw = PropertyAccessorFactory.forBeanPropertyAccess(this);
      ResourceLoader resourceLoader = new ServletContextResourceLoader(getServletContext());
      bw.registerCustomEditor(Resource.class, new ResourceEditor(resourceLoader, getEnvironment()));
      initBeanWrapper(bw);
    //通过BeanWrapper设置DispatchServlet的属性
      bw.setPropertyValues(pvs, true);
  }

  //使子类各自完成初始化
  initServletBean();
  ...
}

注意最后有一个initServletBean()方法,这个 initServletBean() 方法在 HttpSrevletBean 中仅仅声明一下,具体的实现交由子类 FramworkServlet 去实现,而我们的 DispatchServlet 中的 initServletBean()也正是继承自 FramworkServlet的 initServletBean()方法。

  1. 继承自FramworkServlet的initServletBean()方法
protected final void initServletBean() throws ServletException {
  ...
      long startTime = System.currentTimeMillis();

      try {
    //初始化webApplicationContext
          this.webApplicationContext = initWebApplicationContext();
          initFrameworkServlet();
      }
      ...
}

继续深入initWebApplicationContext()方法内部

protected WebApplicationContext initWebApplicationContext() {
  //获取Root WebApplicationContext
  WebApplicationContext rootContext =
              WebApplicationContextUtils.getWebApplicationContext(getServletContext());
  //定义WebApplicationContext对象
      WebApplicationContext wac = null;

  //DispatchServlet有个以WebApplicationContext为参数的构造函数,如果使用以WebApplicationContext为参数的构造函数,则执行这段代码。
      if (this.webApplicationContext != null) {
          wac = this.webApplicationContext;
          if (wac instanceof ConfigurableWebApplicationContext) {
              ConfigurableWebApplicationContext cwac = (ConfigurableWebApplicationContext) wac;
              if (!cwac.isActive()) {
                  if (cwac.getParent() == null) {
                      cwac.setParent(rootContext);
                  }
                  configureAndRefreshWebApplicationContext(cwac);
              }
          }
      }
      if (wac == null) {
    //以contextAttribute属性(FramworkServlet的String类型属性)为Key,从ServletContext中找WebApplicationContext
    //一般不会设置contextAttribute属性,也就是说查找结果一般为null
          wac = findWebApplicationContext();
      }
      if (wac == null) {
          //创建WebApplicationContext
    //后面会深入观察
          wac = createWebApplicationContext(rootContext);
      }

      if (!this.refreshEventReceived) {
          onRefresh(wac);
      }

      if (this.publishContext) {
          // Publish the context as a servlet context attribute.
          String attrName = getServletContextAttributeName();
          getServletContext().setAttribute(attrName, wac);
      }

      return wac;
}

继续深入观察createWebApplicationContext(rootContext)

protected WebApplicationContext createWebApplicationContext(ApplicationContext parent) {
  //找到WebApplicationContext的实现类,默认为XmlWebApplicationContext
  Class contextClass = getContextClass();
      ...

      ConfigurableWebApplicationContext wac =
              (ConfigurableWebApplicationContext) BeanUtils.instantiateClass(contextClass);
  //对wac进行属性设置
      wac.setEnvironment(getEnvironment());
      wac.setParent(parent);
  //getContextConfigLocation()返回"xxx-servlet.xml"
      wac.setConfigLocation(getContextConfigLocation());
  //后面会深入观察
      configureAndRefreshWebApplicationContext(wac);

      return wac;
}

继续深入观察configureAndRefreshWebApplicationContext(wac)

protected void configureAndRefreshWebApplicationContext(ConfigurableWebApplicationContext wac) {
  ...
  //关联servletContext
      wac.setServletContext(getServletContext());
      wac.setServletConfig(getServletConfig());
      wac.setNamespace(getNamespace());
      wac.addApplicationListener(new SourceFilteringListener(wac, new ContextRefreshListener()));

      ConfigurableEnvironment env = wac.getEnvironment();
      if (env instanceof ConfigurableWebEnvironment) {
          ((ConfigurableWebEnvironment) env).initPropertySources(getServletContext(), getServletConfig());
      }

      postProcessWebApplicationContext(wac);
      applyInitializers(wac);
  //刷新WebApplicationContext
      wac.refresh();
}

总之,当refresh()方法执行完毕之后,会触发 继承自FramworkServlet.onApplicationEvent() 函数,该函数会执行内部的 onRefresh()方法。该方法交由子类 DispatchServlet 去实现。如下是 DispatchServlet部分代码。

protected void onRefresh(ApplicationContext context) {
      initStrategies(context);
  }
//通过反射机制查找并装配用户自定义的组件,如果找不到则使用默认的组件进行装配
//默认的装配组件在org.springframework.web.servlet.DispatchServlet,properties文件中定义
  protected void initStrategies(ApplicationContext context) {
  //初始化文件上传解析器
      initMultipartResolver(context);
  //初始化本地化解析器
      initLocaleResolver(context);
  //初始化主体解析器
      initThemeResolver(context);
  //初始化映射
      initHandlerMappings(context);
  //初始化映射适配器
      initHandlerAdapters(context);
  //初始化异常处理器
      initHandlerExceptionResolvers(context);
  //初始化视图名称翻译器
      initRequestToViewNameTranslator(context);
  //初始化视图解析器
      initViewResolvers(context);
  //初始化管理FlashMap的接口,FlashMap用于存储一个请求的输出,当进入另一个请求时作为
  //请求的输入,通常用于重定向场景
      initFlashMapManager(context);
  }

至此,servlet全部初始化完成,就等着第一个请求的到来了,总结为一句话就是:

  • 通过调用init()方法初始化 WebApplicationContext,并通过配置文件配置 WebApplicationContext
  • 初始化各种解析器

你可能感兴趣的:(DispatchServlet初始化源码分析)