DispatcherServlet 是前端控制器设计模式的实现,提供 Spring Web MVC 的集中访问点,而且负责职责的分派,而且与 Spring IoC 容器无缝集成,从而可以获得 Spring 的所有好处。 具体请参考第二章的图 2-1。
DispatcherServlet 主要用作职责调度工作,本身主要用于控制流程,主要职责如下:
1、 文件上传解析,如果请求类型是 multipart 将通过 MultipartResolver 进行文件上传解析;
2、 通过 HandlerMapping,将请求映射到处理器(返回一个 HandlerExecutionChain,它包括一个处理器、多个
HandlerInterceptor 拦截器);
3、 通过 HandlerAdapter 支持多种类型的处理器(HandlerExecutionChain中的处理器);
4、 通过 ViewResolver 解析逻辑视图名到具体视图实现;
5、 本地化解析;
6、 渲染具体的视图等;
7、 如果执行过程中遇到异常将交给 HandlerExceptionResolver 来解析。
从以上我们可以看出 DispatcherServlet 主要负责流程的控制(而且在流程中的每个关键点都是很容易扩展的)。
<servlet>
<servlet-name>chapter2</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>chapter2</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
load-on-startup:表示启动容器时初始化该 Servlet; url-pattern:表示哪些请求交给 Spring Web MVC 处理, “/” 是用来定义默认 servlet 映射的。也可以如“*.html”表示拦截所有以 html 为扩展名的请求。
该 DispatcherServlet 默认使用 WebApplicationContext 作为上下文,Spring 默认配置文件为“/WEB-INF/[servlet名字]-servlet.xml”。
DispatcherServlet也可以配置自己的初始化参数,覆盖默认配置:摘自 Spring Reference
<servlet>
<servlet-name>chapter2</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<load-on-startup>1</load-on-startup>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:spring-servlet-config.xml</param-value>
</init-param>
</servlet>
如果使用如上配置,Spring Web MVC 框架将加载“classpath:spring-servlet-config.xml”来进行初始化上下文而不是“/WEB-INF/[servlet名字]-servlet.xml”。
3.3、上下文关系集成 Web 环境的通用配置:
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>
classpath:spring-common-config.xml, classpath:spring-budget-config.xml
</param-value>
</context-param>
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> </listener>
如上配置是 Spring 集成 Web 环境的通用配置;一般用于加载除 Web 层的 Bean(如 DAO、Service 等),以便于与其他任何 Web 框架集成。
contextConfigLocation:表示用于加载Bean的配置文件; contextClass:表示用于加载Bean的ApplicationContext实现类,默认WebApplicationContext。
ContextLoaderListener初始化的上下文和DispatcherServlet初始化的上下文关系,如图
从图中可以看出:
ContextLoaderListener初始化的上下文加载的Bean是对于整个应用程序共享的,不管是使用什么表现层技术,一般如DAO层、Service层Bean;
DispatcherServlet 初始化的上下文加载的 Bean 是只对 Spring Web MVC 有效的 Bean,如 Controller、 HandlerMapping、HandlerAdapter等等,该初始化上下文应该只加载Web相关组件。
继承体系结构如下所示:
1、 HttpServletBean继承HttpServlet,因此在Web容器启动时将调用它的init方法,该初始化方法的主要作用:::将Servlet初始化参数(init-param)设置到该组件上(如contextAttribute、contextClass、namespace、 contextConfigLocation),通过BeanWrapper简化设值过程,方便后续使用;:::提供给子类初始化扩展点,initServletBean(),该方法由FrameworkServlet覆盖。
2、 FrameworkServlet 继承 HttpServletBean,通过 initServletBean()进行 Web 上下文初始化,该方法主要覆盖一下两件事情: 初始化 web 上下文; 提供给子类初始化扩展点;
3、 DispatcherServlet 继承 FrameworkServlet,并实现了onRefresh()方法提供一些前端控制器相关的配置:
整个 DispatcherServlet 初始化的过程和做了些什么事情,具体主要做了如下两件事情: 1、初始化 Spring Web MVC 使用的 Web 上下文,并且可能指定父容器为(ContextLoaderListener加载了根上下文); 2、初始化 DispatcherServlet 使用的策略,如HandlerMapping、HandlerAdapter等。
DispatcherServlet 的默认配置在 DispatcherServlet.properties(和 DispatcherServlet 类在一个包下)中,而且是当 Spring
配置文件中没有指定配置时使用的默认策略:
DispatcherServlet 默认使用 WebApplicationContext 作为上下文,因此我们来看一下该上下文中有哪些特殊的 Bean: 1、Controller:处理器/页面控制器,做的是 MVC 中的 C 的事情,但控制逻辑转移到前端控制器了,用于对请求进行处理;
2、 HandlerMapping:请求到处理器的映射,如果映射成功返回一个 HandlerExecutionChain 对象(包含一个 Handler 处理器(页面控制器)对象、多个 HandlerInterceptor 拦截器)对象;如BeanNameUrlHandlerMapping将URL与Bean 名字映射,映射成功的Bean就是此处的处理器;
3、 HandlerAdapter:HandlerAdapter 将会把处理器包装为适配器,从而支持多种类型的处理器,即适配器设计模式的应用,从而很容易支持很多类型的处理器;如SimpleControllerHandlerAdapter将对实现了Controller接口的
Bean进行适配,并且掉处理器的handleRequest方法进行功能处理;
4、 ViewResolver:ViewResolver将把逻辑视图名解析为具体的View,通过这种策略模式,很容易更换其他视图技术;如InternalResourceViewResolver将逻辑视图名映射为jsp视图;
5、 LocalResover:本地化解析,因为 Spring 支持国际化,因此 LocalResover 解析客户端的 Locale 信息从而方便进行国际化;
6、 ThemeResovler:主题解析,通过它来实现一个页面多套风格,即常见的类似于软件皮肤效果;
7、 MultipartResolver:文件上传解析,用于支持文件上传;
8、 HandlerExceptionResolver:处理器异常解析,可以将异常映射到相应的统一错误界面,从而显示用户友好的界面(而不是给用户看到具体的错误信息);
9、 RequestToViewNameTranslator:当处理器没有返回逻辑视图名等相关信息时,自动将请求 URL 映射为逻辑视图名;
10、 FlashMapManager:用于管理 FlashMap 的策略接口,FlashMap 用于存储一个请求的输出,当进入另一个请求时作为该请求的输入,通常用于重定向场景,后边会细述。
到此 DispatcherServlet 我们已经了解了,接下来我们就需要把上边提到的特殊 Bean 挨个击破,那首先从控制器开始吧。
说明:博客原文