谈谈你对Spring MVC中的九大组件的理解?

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

一位应届毕业生被问到这样一道面试题,说谈谈你对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 多文件上传组件。

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

<form method="post" enctype="multipart/form-data" >
<input type="file"/>
<input type="file"/>
<input type="file"/>
<button type="submit">上传</button>
</form>

主要逻辑就是将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<HandlerMapping> handlerMappings;  
private void initHandlerMappings(ApplicationContext context) {
this.handlerMappings = null;
if (this.detectAllHandlerMappings) {
Map<String, HandlerMapping> 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只是对模板文件的封装,它是用作参数来传递。

责任编辑:姜华 来源: 今日头条
相关推荐

2022-06-30 09:10:33

NoSQLHBaseRedis

2022-01-05 08:53:13

Spring原理分析MVC

2022-09-06 11:13:16

接口PipelineHandler

2022-09-19 07:57:59

云服务互联网基础设施

2022-08-14 07:14:50

Kafka零拷贝

2022-10-09 15:18:31

SwaggerOpenAPI工具

2022-09-09 10:15:06

OAuthJava

2022-03-21 09:05:18

volatileCPUJava

2022-08-26 00:21:44

IO模型线程

2022-09-23 11:00:27

KafkaZookeeper机制

2022-08-26 00:02:03

RocketMQ单体架构MQ

2022-06-10 11:51:49

MySQL事务隔离

2009-04-24 09:43:09

.NETASP.NET框架

2022-08-29 16:03:33

状态流转Java

2019-07-26 06:42:28

PG架构数据库

2017-06-02 09:47:29

网络分层协议

2022-01-04 20:52:50

函数异步Promise

2022-02-10 14:38:28

前端框架浏览器

2021-09-12 22:22:15

前端

2019-05-10 10:50:04

Spring AOPJDK动态代理CGLIB动态代理
点赞
收藏

51CTO技术栈公众号