SpringMVC工作原理详解

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析DNS,个人版 1个月
云解析 DNS,旗舰版 1个月
简介: SpringMVC框架是以请求为驱动,围绕Servlet设计,将请求发给控制器,然后通过模型对象,分派器来展示请求结果视图。其中核心类是DispatcherServlet,它是一个Servlet,顶层实现的是Servlet接口。

1、SpringMVC中DispatcherServlet的介绍

SpringMVC框架是以请求为驱动,围绕Servlet设计,将请求发给控制器,然后通过模型对象,分派器来展示请求结果视图。其中核心类是DispatcherServlet,它是一个Servlet,顶层实现的是Servlet接口。

2、SpringMVC框架的配置

需要在web.xml中配置DispatcherServlet。并且需要配置Spring的监听器ContextLoaderListener

配置代码如下:

<listener>
            <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
        </listener>        
        <servlet>
  <servlet-name>springmvc</servlet-name>
  <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
                <!-- 如果不设置init-param标签,则必须在/WEB-INF/下创建xxx-servlet.xml文件,其中xxx是servlet-name中配置的名称。  -->
                <init-param>
    <param-name>contextConfigLocation</param-name>
    <param-value>classpath:spring/springmvc-servlet.xml</param-value>
  </init-param>
  <load-on-startup>1</load-on-startup>
  </servlet>
  <servlet-mapping>
  <servlet-name>springmvc</servlet-name>
  <url-pattern>/</url-pattern>
  </servlet-mapping>

3、SpringMVC运行原理图

如下:

e68d0b79ffbce35a68554bfb6fa1f10b_watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3lhbmdfZ3Vhbmcz,size_16,color_FFFFFF,t_70.png


流程说明:


(1)客户端(浏览器)发出http请求给web服务器,服务器对请求进行解析,如果匹配到DispatcherServlet的请求映射路径,然后将请求转交给DispatcherServlet;


(2)DispatcherServlet根据请求信息(包括URL、Http方法、请求报文头和请求参数Cookie等)调用HandlerMapping,解析请求对应的Handler;


(3)解析到对应的Handler后,开始由HandlerAdapter适配器处理。


(4)HandlerAdapter会根据Handler来调用真正的处理器去处理请求,并处理相应的业务逻辑。


(5)处理器处理完业务后,会返回一个ModelAndView对象,Model是返回的数据对象,View是个逻辑上的视图。


(6)ViewResolver会根据逻辑View找到真正的视图。


(7)DispaterServlet通过model对ModelAndView()中的参数进行解析,最终展现出完整的View并返回给客户端。

4、DispatcherServlet详细解析

首先看下源码:

package org.springframework.web.servlet;
@SuppressWarnings("serial")
public class DispatcherServlet extends FrameworkServlet {
  public static final String MULTIPART_RESOLVER_BEAN_NAME = "multipartResolver";
  public static final String LOCALE_RESOLVER_BEAN_NAME = "localeResolver";
  public static final String THEME_RESOLVER_BEAN_NAME = "themeResolver";
  public static final String HANDLER_MAPPING_BEAN_NAME = "handlerMapping";
  public static final String HANDLER_ADAPTER_BEAN_NAME = "handlerAdapter";
  public static final String HANDLER_EXCEPTION_RESOLVER_BEAN_NAME = "handlerExceptionResolver";
  public static final String REQUEST_TO_VIEW_NAME_TRANSLATOR_BEAN_NAME = "viewNameTranslator";
  public static final String VIEW_RESOLVER_BEAN_NAME = "viewResolver";
  public static final String FLASH_MAP_MANAGER_BEAN_NAME = "flashMapManager";
  public static final String WEB_APPLICATION_CONTEXT_ATTRIBUTE = DispatcherServlet.class.getName() + ".CONTEXT";
  public static final String LOCALE_RESOLVER_ATTRIBUTE = DispatcherServlet.class.getName() + ".LOCALE_RESOLVER";
  public static final String THEME_RESOLVER_ATTRIBUTE = DispatcherServlet.class.getName() + ".THEME_RESOLVER";
  public static final String THEME_SOURCE_ATTRIBUTE = DispatcherServlet.class.getName() + ".THEME_SOURCE";
  public static final String INPUT_FLASH_MAP_ATTRIBUTE = DispatcherServlet.class.getName() + ".INPUT_FLASH_MAP";
  public static final String OUTPUT_FLASH_MAP_ATTRIBUTE = DispatcherServlet.class.getName() + ".OUTPUT_FLASH_MAP";
  public static final String FLASH_MAP_MANAGER_ATTRIBUTE = DispatcherServlet.class.getName() + ".FLASH_MAP_MANAGER";
  public static final String EXCEPTION_ATTRIBUTE = DispatcherServlet.class.getName() + ".EXCEPTION";
  public static final String PAGE_NOT_FOUND_LOG_CATEGORY = "org.springframework.web.servlet.PageNotFound";
  private static final String DEFAULT_STRATEGIES_PATH = "DispatcherServlet.properties";
  protected static final Log pageNotFoundLogger = LogFactory.getLog(PAGE_NOT_FOUND_LOG_CATEGORY);
  private static final Properties defaultStrategies;
  static {
  try {
    ClassPathResource resource = new ClassPathResource(DEFAULT_STRATEGIES_PATH, DispatcherServlet.class);
    defaultStrategies = PropertiesLoaderUtils.loadProperties(resource);
  }
  catch (IOException ex) {
    throw new IllegalStateException("Could not load 'DispatcherServlet.properties': " + ex.getMessage());
  }
  }
  /** Detect all HandlerMappings or just expect "handlerMapping" bean? */
  private boolean detectAllHandlerMappings = true;
  /** Detect all HandlerAdapters or just expect "handlerAdapter" bean? */
  private boolean detectAllHandlerAdapters = true;
  /** Detect all HandlerExceptionResolvers or just expect "handlerExceptionResolver" bean? */
  private boolean detectAllHandlerExceptionResolvers = true;
  /** Detect all ViewResolvers or just expect "viewResolver" bean? */
  private boolean detectAllViewResolvers = true;
  /** Throw a NoHandlerFoundException if no Handler was found to process this request? **/
  private boolean throwExceptionIfNoHandlerFound = false;
  /** Perform cleanup of request attributes after include request? */
  private boolean cleanupAfterInclude = true;
  /** MultipartResolver used by this servlet */
  private MultipartResolver multipartResolver;
  /** LocaleResolver used by this servlet */
  private LocaleResolver localeResolver;
  /** ThemeResolver used by this servlet */
  private ThemeResolver themeResolver;
  /** List of HandlerMappings used by this servlet */
  private List<HandlerMapping> handlerMappings;
  /** List of HandlerAdapters used by this servlet */
  private List<HandlerAdapter> handlerAdapters;
  /** List of HandlerExceptionResolvers used by this servlet */
  private List<HandlerExceptionResolver> handlerExceptionResolvers;
  /** RequestToViewNameTranslator used by this servlet */
  private RequestToViewNameTranslator viewNameTranslator;
  private FlashMapManager flashMapManager;
  /** List of ViewResolvers used by this servlet */
  private List<ViewResolver> viewResolvers;
  public DispatcherServlet() {
  super();
  }
  public DispatcherServlet(WebApplicationContext webApplicationContext) {
  super(webApplicationContext);
  }
  @Override
  protected void onRefresh(ApplicationContext context) {
  initStrategies(context);
  }
  protected void initStrategies(ApplicationContext context) {
  initMultipartResolver(context);
  initLocaleResolver(context);
  initThemeResolver(context);
  initHandlerMappings(context);
  initHandlerAdapters(context);
  initHandlerExceptionResolvers(context);
  initRequestToViewNameTranslator(context);
  initViewResolvers(context);
  initFlashMapManager(context);
  }
}

DispatcherServlet类中的属性:


HandlerMapping:用于handlers映射请求和一系列的对于拦截器的前处理和后处理,大部分用@Controller注解。


HandlerAdapter:帮助DispatcherServlet处理映射请求处理程序的适配器,而不用考虑实际调用的是哪个处理程序。


HandlerExceptionResolver:处理映射异常。


ViewResolver:根据实际配置解析实际的View类型。


LocaleResolver:解决客户正在使用的区域设置以及可能的时区,以便能够提供国际化视野。


ThemeResolver:解决Web应用程序可以使用的主题,例如提供个性化布局。


MultipartResolver:解析多部分请求,以支持从HTML表单上传文件。


FlashMapManager:存储并检索可用于将一个请求属性传递到另一个请求的input和output的FlashMap,通常用于重定向。


在Web MVC框架中,每个DispatcherServlet都拥自己的WebApplicationContext,它继承了ApplicationContext。WebApplicationContext包含了其上下文和Servlet实例之间共享的所有的基础框架beans。


其中几个重要属性的详细解释:


HandlerMapping:

600c3d8199065bf81773659fbef7e735_watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3lhbmdfZ3Vhbmcz,size_16,color_FFFFFF,t_70.png

HandlerMapping接口处理请求的映射


HandlerMapping接口的实现类:


SimpleUrlHandlerMapping类通过配置文件把URL映射到Controller类。


HandlerAdapter:

34ce3751d288799b4050ba143bee362b_watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3lhbmdfZ3Vhbmcz,size_16,color_FFFFFF,t_70.png

HandlerAdapter接口-处理请求映射


AnnotationMethodHandlerAdapter:通过注解,把请求URL映射到Controller类的方法上。


HandlerExceptionResolver:

cd63eea3b4b59b6b8dbd86030a657a0f_watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3lhbmdfZ3Vhbmcz,size_16,color_FFFFFF,t_70.png

HandlerExceptionResolver接口-异常处理接口


SimpleMappingExceptionResolver通过配置文件进行异常处理。


AnnotationMethodHandlerExceptionResolver:通过注解进行异常处理。


ViewResolver:

a33e2a9eb3473015b7d8bd3344de3936_watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3lhbmdfZ3Vhbmcz,size_16,color_FFFFFF,t_70.png


ViewResolver接口解析View视图。


UrlBasedViewResolver类 通过配置文件,把一个视图名交给到一个View来处理。


目录
相关文章
|
3月前
|
设计模式 开发框架 前端开发
SpringMVC原理分析 | Hello程序
SpringMVC原理分析 | Hello程序
34 0
|
设计模式 XML JSON
springMVC 工作原理和流程
springMVC 工作原理和流程
83 0
springMVC 工作原理和流程
|
存储 前端开发 Java
springMvc简要总结
springMvc简要总结
111 0
|
XML 设计模式 前端开发
springMvc43-springMVC运行原理
springMvc43-springMVC运行原理
101 0
springMvc43-springMVC运行原理
|
设计模式 XML 前端开发
Springmvc:框架原理
Springmvc:框架原理
103 0
Springmvc:框架原理
|
开发框架 前端开发 JavaScript
01Springmvc的基本概念
本文讲述了SpringMVC的基本概念
165 0
|
设计模式 JSON 前端开发
springmvc(一) springmvc框架原理分析和简单入门程序
springmvc这个框架真的非常简单,感觉比struts2还更简单,好好沉淀下来学习~
355 0
springmvc(一) springmvc框架原理分析和简单入门程序
SpringMVC源码分析6:SpringMVC的视图解析原理
SpringMVC视图机制详解[附带源码分析] 目录 前言 重要接口和类介绍 源码分析 编码自定义的ViewResolver 总结 参考资料 前言 SpringMVC是目前主流的Web MVC框架之一。
|
前端开发
SpingMVC的工作原理,就是这么简单
用了这么久的SpingMVC,她的工作流程还是要搞清楚。 一、springmvc流程先上一张手绘流程图吧, SpingMVC的工作原理,就是这么简单1、客户端发起请求到前端控制器(DispatcherServlet); 2、前端控制器请求处理器映射器(HandlerMappering)通过xml配...
1351 0