在基于Spring框架的Web应用中,Spring的应用上下文配置文件applicationgContext.xml
是如何自动加载的?
当运行一个Web项目时,应用服务器(JBoss、Tomcat等)首先会读取项目源码路径中的web.xml
文件,解析其中的配置,发现配置了ContextLoaderListener
,因此会执行ContextLoaderListener
类中的contextInitialized
方法,在这个方法中会调用initWebApplicationContext()
方法,根据方法名可以看出这个方法是用于初始化一个WebApplicationContext,简单理解就是初始化一个Web应用下的Spring容器。在initWebApplicationContext()
方法后续代码实现的内部会根据web.xml
中配置的contextConfigLocation
属性加载指定的applicationContext.xml
文件,根据这个文件初始化Spring容器。
如果在web.xml
中没有配置contextConfigLocation
参数,是不是就不能加载applicationgContext.xml
文件?
如果没有配置contextConfigLocation
参数,那么应用启动时会默认查找应用根目录下/WEB-INF/applicationContext.xml
文件,也就是说这是一个默认加载的文件路径。
这个Web应用下的WebApplicationContext初始化完成后,它和ServletContext有什么关系呢?
在initWebApplicationContext
方法的内部会把初始化后的context
存到ServletContext
中,具体的就是存到了一个Map
类型的属性中,key
是WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE
,value
是具体的WebApplicationContext实例对象。
如果想在代码中通过ServletContext
获取这个WebApplicationContext
,如何操作?
Spring框架提供了一个WebApplicationContextUtils
工具类,通过这个工具类的getWebApplicationContext
方法可以获取到。
刚刚说到ServletContext
,它到底做什么用的?
ServletContext
定义了一些方法方便Servlet
和Servlet
容器进行通讯,在一个web应用中所有的Servlet
都公用一个ServletContext
,Spring在和web应用结合使用的时候,是将Spring的容器存到ServletContext
中的,通俗的说就是将一个ApplicationContext
存储到ServletContext
的一个Map属性中。
有了解过web.xml
中Listener
、Filter
和Servlet
的初始化顺序吗?
首先,对使用<listener>
标签声明的监听器类进行实例化,调用监听器类实例对象的contextInitialized()
方法,初始化应用上下文数据;接着对使用<filter>
标签声明的过滤器类进行实例化,调用过滤器类实例对象的init()
方法;如果<servlet>
标签内使用了<load-on-startup>
标签,则按照数值从小到大的顺序对Servlet
进行实例化,并调用对应的init()
方法。
说到Servlet
,Spring MVC中的DispatcherServlet
有了解过吗?说一下它的实现原理?
DispatcherServlet
是SpringMVC的核心分发器,它实现了请求分发,是处理请求的入口。它是一个Servlet
,在应用启动时,DispatcherServlet
初始化会执行init
方法,查看源码发现DispatcherServlet
的init
方法继承自HttpServletBean
,在这个初始化方法中会实例化一个WebApplicationContext
对象,并且将初始化后的context
存到ServletContext
中,让Servlet
和Spring容器进行关联。在DispatcherServlet
的onRefresh
方法中,初始化各种请求处理策略,例如文件上传处理策略、URL请求处理策略、视图映射处理策略、异常处理策略等,这些策略的大部分执行逻辑都是先从WebApplicationContext
中查找,找不到的情况下再加载和DispatcherServlet
同目录下的DispatcherServlet.properties
中的各个策略,例如初始化HandlerMapping
,注册各种请求的处理策略及处理类。
具体说一下DispatcherServlet
请求分发的实现原理?
首先,Spring MVC框架在启动的时候会遍历Spring容器中的所有bean,对标注了@Controller
或@RequestMapping
注解的类中方法进行遍历,将类和方法上的@RequestMapping
注解值进行合并,使用@RequestMapping
注解的相关参数值(如value
、method
等)封装一个RequestMappingInfo
,将这个Controller
实例、方法及方法参数信息(类型、注解等)封装到HandlerMethod
中,然后以RequestMappingInfo
为key
,HandlerMethod
为value
存到一个以Map
为结构的handlerMethods
中。
接着,将@RequestMapping
注解中的value
(即请求路径)值取出,即url
,然后以url
为key
,以RequestMappingInfo
为value
,存到一个以Map
为结构的urlMap
属性中。
客户端发起请求的时候,根据请求的URL
到urlMap
中查找,找到RequestMappingInfo
,然后根据RequestMappingInfo
到handlerMethods
中查找,找到对应的HandlerMethod
,接着将HandlerMethod
封装到HandlerExecutionChain
;接着遍历容器中所有HandlerAdapter
实现类,找到支持这次请求的HandlerAdapter
,如RequestMappingHandlerAdapter
,然后执行Spring MVC拦截器的前置方法(preHandle
方法),然后对请求参数解析及转换,然后(使用反射)调用具体Controller
的对应方法返回一个ModelAndView
对象,执行拦截器的后置方法(postHandle
方法),然后对返回的结果进行处理,最后执行afterCompletion
方法。