一位应届毕业生被问到这样一道面试题,说谈谈你对Spring MVC中的九大组件的理解。

今天,我给大家分享一下我的理解。

1、Spring MVC九大组件

使用Spring MVC框架时,它的主要入口是DispatcherServlet类,Spring MVC子容器初始化时,会调用DispatcherServlet的onRefresh()方法,而onRefresh()方法只做了一件事,就是调用initStrategies()方法来初始化Spring MVC的九大组件,如源码所示:


(资料图片仅供参考)

protected void onRefresh(ApplicationContext context) {      this.initStrategies(context);}protected void initStrategies(ApplicationContext context) {      this.initMultipartResolver(context);      this.initLocaleResolver(context);      this.initThemeResolver(context);      this.initHandlerMappings(context);      this.initHandlerAdapters(context);      this.initHandlerExceptionResolvers(context);      this.initRequestToViewNameTranslator(context);      this.initViewResolvers(context);      this.initFlashMapManager(context);}

顾名思义,initStrategies()方法直译过来就是初始化策略,Spring MVC把九大组件设计成九大策略,其实就是为了明确各个组件的职责,达到解耦的目的。

Spring MVC的九大组件按照初始化顺序分别为:MultipartResolver多文件上传组件、LocaleResolver多语言支持组件、ThemeResolver主题模板处理组件、HandlerMappings URL映射组件、HandlerAdapters业务逻辑适配组件、HandlerExceptionResolvers异常处理组件、RequestToViewNameTranslator视图名称提取组件、ViewResolvers视图渲染组件和FlashMapManager闪存管理组件。

下面给大家详细分析一下,每个组件的功能和职责。

1、MultipartResolver 多文件上传组件。

用于支持多文件上传,如代码所示:

主要逻辑就是将enctype为"multipart/form-data"的表单request请求包装成MultipartHttpServletRequest。程序员在开发的时候,只需要调用MultipartHttpServletRequest的 getFile()方法,就可以获取客户端上传的文件列表了。

2、LocaleResolver多语言支持组件。

用于支持国际化多语言切换,LocaleResolver的主要作用就是从 request 中解析出 local 参数的值,如源码所示:

public interface LocaleResolver {      Locale resolveLocale(HttpServletRequest request);      void setLocale(HttpServletRequest request, HttpServletResponse response, Locale local);}}

resolveLocale()方法是从 request 中解析出 local,setLocale()方法是将指定的 local 值设置到 request 中。而 local 大多数情况下都是用来做国际化处理的,配合多语言字典properties来使用,例如中国的Local值为zh_CN。

3、ThemeResolver主题模板处理组件。

主要用于支持Web页面的多主题风格。可以通过ThemeResolver来读取和解析页面主题样式配置。实现原理和LocaleResolver类似,也是配置一套 properties 文件,根据不同参数来切换读取;当然,使用ThemeResolver也是可以实现国际化。如源代码所示:

public interface ThemeResolver {    String resolveThemeName(HttpServletRequest request);    void setThemeName(HttpServletRequest request, HttpServletResponse response, String themeName);}

它的主要方法也和LocaleResolver类似,一个从request提取主题名称的方法,一个设定主题名称的方法。

4、HandlerMappingURL映射组件。

主要是用来保存Url和业务逻辑的对应关系,它本质上就是一个Map,Key为URL值就是对应Controller中配置了@RequestMapping注解的方法。但是在Spring源码中,被封装成了一个HandlerMapping对象。然后,每个HandlerMapping对象都被缓存在一个List中。如源码所示:

private List handlerMappings;  private void initHandlerMappings(ApplicationContext context) {   this.handlerMappings = null;   if (this.detectAllHandlerMappings) {      Map matchingBeans =            BeanFactoryUtils.beansOfTypeIncludingAncestors(context, HandlerMapping.class, true, false);      if (!matchingBeans.isEmpty()) {         this.handlerMappings = new ArrayList<>(matchingBeans.values());         AnnotationAwareOrderComparator.sort(this.handlerMappings);      }   }   ...}
5、HandlerAdapter业务逻辑适配组件。

主要功能是动态解析参数以及动态适配业务逻辑对应的Handler,如源码所示:

public interface HandlerAdapter {      boolean supports(Object handler);      ModelAndView handle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception;      long getLastModified(HttpServletRequest request, Object handler);}

在HandlerAdapter中提供了一个叫做handle()这样一个方法,第三个参数 Object handler 第三个参数其实就是业务处理器。在DispatcherServlet中的doDispath()方法中被调用。而handler对象就是根据用户请求的Url从HandlerMapping获取的HandlerMapiping对象。

在HandlerApdater的handle()方法中,首先会动态解析用户传过来的参数,并完成数据类型转化。然后,反射调用HandlerMapiping封装的Controller中的方法,最后,将调用方法的返回结果统一封装为ModelAndView。

6、HandlerExceptionResolver异常处理组件。

主要用于拦截对不同异常的个性化处理,Spring可以给不同的异常配置不同的ModelAndView,HandlerExceptionResolver根据异常类型,的将处理封装为一个ModelAndView从而将异常信息转换为更加友好的Web页面展示,如源码所示:

public interface HandlerExceptionResolver {      ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex);}

HandlerExceptionResolver组件只有一个方法,就是将异常解析为ModelAndView。

当然,HandlerExceptionResolver自己发生异常或者在异常页面渲染过程中发生异常HandlerExceptionResolver不会处理。Spring可以配置一个全局的500页面或者404页面来处理这个问题。

7、RequestToViewNameTranslator视图名称提取组件。

这个组件的主要功能是可从request中提取viewName。这个viewName设置在url参数上,也可以设置在request的header上。如源码所示:

public interface RequestToViewNameTranslator {      String getViewName(HttpServletRequest request) throws Exception;}

这个其实还是挺有意思的,就是将 request 请求转换为视图名称。它只有一个getViewName()方法。

8、ViewResolvers视图渲染组件。

它的作用相当于模板引擎,就是根据视图名称找到视图对应的模板文件,然后进行解析,如源码所示:

public interface ViewResolver {      View resolveViewName(String viewName, Locale local) throws Exception;}

ViewResolvers组件只有一个resolveViewName()方法,

我们看到resolveViewName()方法有两个参数。

第一个参数viewName,是String类型,它其实就是视图名称,对应的就是模板文件的名称。第二个参数local,前面我们讲过代表的是本地语言环境,可以用来做国际化。

resolveViewName()方法的放回值是一个View对象。而View对象就是用来渲染页面的,也就是说将程序返回的结果填入到具体的模板里面,生成具体的视图文件,比如:jsp,ftl,html 等等。

9、FlashMapManage闪存管理组件。

它相当于一个参数缓存器,用来保证请求跳转过程中参数不丢失,和Struts 2中的ValueStack值栈非常类似。主要是 redirect重定向的时候,参数传递会丢失,FlashMapManage就能大显身手,可以做到Redirect重定向和Forward转发同样的效果,如源码所示:

public interface FlashMapManager {    FlashMap retrieveAndUpdate(HttpServletRequest request, HttpServletResponse response);   void saveOutputFlashMap(FlashMap flashMap, HttpServletRequest request, HttpServletResponse response);}

FlashMapManage主要有两个方法,

retrieveAndUpdate()方法是用来恢复参数的,而且对于恢复过的和超时的参数都会被删除掉。saveOutputFlashMap() 这个方法是用来保存参数的。

FlashMapManager默认会将参数保存在 Session 中,在日常开发中,如果不想将参数暴露在 Url路径中,那就可以在请求转发时,在参数中添加@RedirectAttributes注解将参数缓存,然后在下一个处理器中就可以获取到。

以上就是我对Spring MVC中的九大组件的理解。

需要注意的是ModelAndView和View并不属于MVC的九大组件之中,ModelAndView只是对ViewName和Model的封装,然后作为返回值把信息反馈给用户。并没有包含任何执行逻辑。而View只是对模板文件的封装,它是用作参数来传递。

推荐内容