SpringMVC源码解析 - HandlerAdater - ModelAndViewContainer上下文容器
HandlerAdapter在处理请求时上下文数据的传递工作是由ModelAndViewContainer负责的.
源码注释是这样描述的:
Records model and view related decisions made by HandlerMethodArgumentResolvers and HandlerMethodReturnValueHandlers during the course of invocation of a controller method.
翻译下: 记录HandlerMethodArgumentResolver 和 HandlerMethodReturnValueHandler 在处理handler时 使用的模型model和视图view相关信息.
ModelAndViewContainer主要职责:
1. 维护模型model,包括defaultModle和redirectModel
2. 维护视图view
3. 维护是否redirect信息,及根据这个判断HandlerAdapter使用的是defaultModel或redirectModel(判断规则详见下文)
4. 维护@SessionAttributes注解信息状态
5. 维护handler是否处理标记
目录:
1. ModelAndViewContainer属性
2. 科普ModelMap继承体系
3. ModelAndView提供的api,复杂的还是model相关的属性设置,其他主要是简单的getter,setter
ModelAndViewContainer属性
先来看看ModelAndViewContainer的属性,这样就比较清晰:
package org.springframework.web.method.support;
public class ModelAndViewContainer {
// 视图,实际使用时可能是String类型的逻辑视图
private Object view;
// 标记handler是否已经完成请求处理
private boolean requestHandled = false;
// 默认模型,下文我们可以简单科普下ModelMap继承体系
private final ModelMap defaultModel = new BindingAwareModelMap();
// redirect时使用的模型,实际使用的是RedirectAttributesModelMap
private ModelMap redirectModel;
// 标记处理器返回redirect视图
private boolean redirectModelScenario = false;
// redirect时,是否忽略defaultModel
private boolean ignoreDefaultModelOnRedirect = false;
// @SessionAttributes注解使用状态标记,就是是否处理完毕
private final SessionStatus sessionStatus = new SimpleSessionStatus();
}
顺便学个英语单词,Scenario 方案

科普ModelMap继承体系
继续往下分析之前,我们先来简单科普下ModelMap的继承体系:

各个类的职责与使用场景:
ModelMap是LinkedHashMap的子类,主要是封装attribute概念,实际处理还是委托给map
ExtendedModelMap添加链调用chained calls,并实现Model接口
BindingAwareModelMap,BindingResult相关属性被设置时,自动清除BindingResult.defaultModel使用的该类.
RedirectAttributesModelMap 通过DataBinder做参数类型转换,redirect时使用的flash attributes
各有侧重地看下源码吧
2.1 ModelMap,是继承LinkedHashMap,添加mergeAttributes
package org.springframework.ui;
public class ModelMap extends LinkedHashMap<String, Object> {
public ModelMap mergeAttributes(Map<String, ?> attributes) {
if (attributes != null) {
for (String key : attributes.keySet()) {
if (!containsKey(key)) {
put(key, attributes.get(key));
}
}
}
return this;
}
// ...
}
2.2 ExtendedModelMap添加链调用chained calls
package org.springframework.ui;
public class ExtendedModelMap extends ModelMap implements Model {
@Override
public ExtendedModelMap addAttribute(String attributeName, Object attributeValue) {
super.addAttribute(attributeName, attributeValue);
return this;// 看这里
}
// ...
}
2.3 BindingAwareModelMap,BindingResult相关属性被设置时,自动清除BindingResult.defaultModel使用的该类.
看的就是removeBindingResultIfNecessary.这边put和putAll都会调用removeBindingResultIfNecessary
package org.springframework.validation.support;
public class BindingAwareModelMap extends ExtendedModelMap { @Override
public Object put(String key, Object value) {
removeBindingResultIfNecessary(key, value);
return super.put(key, value);
}
// ...
private void removeBindingResultIfNecessary(Object key, Object value) {
if (key instanceof String) {
String attributeName = (String) key;
if (!attributeName.startsWith(BindingResult.MODEL_KEY_PREFIX)) {
String bindingResultKey = BindingResult.MODEL_KEY_PREFIX + attributeName;
BindingResult bindingResult = (BindingResult) get(bindingResultKey);
if (bindingResult != null && bindingResult.getTarget() != value) {
remove(bindingResultKey);
}
}
}
} }
2.4 RedirectAttributesModelMap 通过DataBinder做参数类型转换,redirect时使用的flash attributes
这样主要就是添加了dataBinder和flashAttributes属性相关的api
package org.springframework.web.servlet.mvc.support;
public class RedirectAttributesModelMap extends ModelMap implements RedirectAttributes { 4 private final DataBinder dataBinder;
6 private final ModelMap flashAttributes = new ModelMap();
private String formatValue(Object value) {
if (value == null) {
return null;
}
12 return (dataBinder != null) ? dataBinder.convertIfNecessary(value, String.class) : value.toString();
}
// ...
}
ModelAndView提供的api
主要是两类api,view相关和model相关.
3.1 view相关其实就一个api,是否类应用(通过是否是string类型判断):
package org.springframework.web.method.support;
public class ModelAndViewContainer {
// ...
/**
* Whether the view is a view reference specified via a name to be
* resolved by the DispatcherServlet via a ViewResolver.
*/
public boolean isViewReference() {
return (this.view instanceof String);
}
}
3.2 model相关的api就比较多了,主要是设置模型值.
在设置模型值的时候,这边涉及到一个问题,就是container里有两个model,往哪个里设置?
container索性抽象一个getModel()的api进行判断.
我们来说说判断的逻辑吧:
使用defaultModel的场景:
a,不是redirect
b,redirect场景下,redirectModel为null,同时ignoreDefaultModelOnRedirect为false
剩下的就是使用redirectModel的场景了,不细说
model相关属性操作的类都是跟addAttribute类似的逻辑,使用getModel获取model后直接委托.所以篇幅缘故下面的源码以addAttribute为例,其他都只是展现api 的signature
package org.springframework.web.method.support;
public class ModelAndViewContainer {
// ...
/**
* Return the model to use: the "default" or the "redirect" model.
* <p>The default model is used if {@code "redirectModelScenario=false"} or
* if the redirect model is {@code null} (i.e. it wasn't declared as a
* method argument) and {@code ignoreDefaultModelOnRedirect=false}.
*/
public ModelMap getModel() {
if (useDefaultModel()) {
return this.defaultModel;
}
else {
return (this.redirectModel != null) ? this.redirectModel : new ModelMap();
}
}
/**
* Whether to use the default model or the redirect model.
* 不是redirect || redirect时redirectModel为空同时不忽略defaultModel
*/
private boolean useDefaultModel() {
return !this.redirectModelScenario || ((this.redirectModel == null) && !this.ignoreDefaultModelOnRedirect);
}
public ModelAndViewContainer addAttribute(String name, Object value) {
getModel().addAttribute(name, value);
return this;
}
public ModelAndViewContainer addAttribute(Object value){};
public ModelAndViewContainer addAllAttributes(Map<String, ?> attributes){};
public ModelAndViewContainer removeAttributes(Map<String, ?> attributes){};
public boolean containsAttribute(String name){};
}
SpringMVC源码解析 - HandlerAdater - ModelAndViewContainer上下文容器的更多相关文章
- SpringMVC源码解析- HandlerAdapter - ModelFactory(转)
ModelFactory主要是两个职责: 1. 初始化model 2. 处理器执行后将modle中相应参数设置到SessionAttributes中 我们来看看具体的处理逻辑(直接充当分析目录): 1 ...
- SpringMVC源码解析- HandlerAdapter - ModelFactory
ModelFactory主要是两个职责: 1. 初始化model 2. 处理器执行后将modle中相应参数设置到SessionAttributes中 我们来看看具体的处理逻辑(直接充当分析目录): 1 ...
- springMVC源码解析--ViewResolver视图解析器执行(三)
之前两篇博客springMVC源码分析--ViewResolver视图解析器(一)和springMVC源码解析--ViewResolverComposite视图解析器集合(二)中我们已经简单介绍了一些 ...
- springMVC源码解析--HandlerMethodArgumentResolverComposite参数解析器集合(二)
上一篇博客springMVC源码分析--HandlerMethodArgumentResolver参数解析器(一)中我们已经介绍了参数解析相关的东西,并且也提到了HandlerMethodArgume ...
- SpringMVC源码解析- HandlerAdapter初始化
HandlerAdapter初始化时,主要是进行注解解析器初始化注册;返回值处理类初始化;全局注解@ControllerAdvice内容读取并缓存. 目录: 注解解析器初始化注册:@ModelAttr ...
- SpringMVC源码解析
一:springmvc运行过程: 1. dispatcherServlet 通过 HandlerMapping 找到controller2. controller经过后台逻辑处理得到结果集modela ...
- 深入了解SpringMVC源码解析
Spring MVC源码解析 Spring MVC的使用原理其实是通过配置一个Servlet来接管所有的请求,所有的请求由这个Servlet来进行分发处理. 我们可以从web.xml里面看出这一点 & ...
- springMVC源码解析--ViewResolverComposite视图解析器集合(二)
上一篇博客springMVC源码分析--ViewResolver视图解析器(一)中我们介绍了一些springMVC提供的很多视图解析器ViewResolver,在开发的一套springMVC系统中是可 ...
- springmvc源码解析MvcNamespaceHandler之<mvc:view-resolvers>
说在前面 本次主要介绍springmvc配置解析. springmvc配置解析 本次介绍MvcNamespaceHandler. 进入到这个方法org.springframework.web.serv ...
随机推荐
- linux中内核延时函数 (转)
第一类延时函数原型是:(忙等) void ndelay(unsigned long nsecs); void udelay(unsigned long usecs); void mdelay(unsi ...
- C# List的深复制
1.关于深拷贝和浅拷贝 C#支持两种类型:值类型和引用类型 值类型(Value Type):如 char, int, float,枚举类型和结构类型 引用类型(Reference Type):如Cla ...
- 设置vim颜色方案
获取所有vim颜色配置方案 ls /usr/share/vim/vim74/colors/ [root@lx ~]# ls /usr/share/vim/vim74/colors/ blue.vim ...
- NOIP2013 Day2
1.积木大赛 https://www.luogu.org/problemnew/show/1969 这道题在考试时暴力得比较麻烦,导致只得了80分,t了两个点. 思路为寻找一个区间内高度大于0的最低点 ...
- 用Linq对String类型的数字求和
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.T ...
- Java 正则表达式详细实例解析
案例1:判断字符串是否是abc package Regex; import java.util.regex.Matcher; import java.util.regex.Pattern; /** * ...
- 学习笔记之C# 教程 | 菜鸟教程
C# 教程 | 菜鸟教程 http://www.runoob.com/csharp/csharp-tutorial.html 菜鸟教程在线编辑器 http://www.runoob.com/try/r ...
- springboot中对yaml文件的解析
一.YAML是“YAML不是一种标记语言”的外语缩写 (见前方参考资料原文内容):但为了强调这种语言以数据做为中心,而不是以置标语言为重点,而用返璞词重新命名.它是一种直观的能够被电脑识别的数据序列化 ...
- javascript 判断空数组
javascript里判断空数组不能用 []==[] 这样来判断,因为数组也是个对象,普通对象通过指针指向的内存中的地址来做比较 所以 []==[]结果为false,因此判断数组是否为空 用 [].l ...
- Struts2单例和多例
struts2中action是多例的,即一个session产生一个action如果是单例的话,若出现两个用户都修改一个对象的属性值,则会因为用户修改时间不同,两个用户访问得到的属性不一样,操作得出的结 ...