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上下文容器的更多相关文章

  1. SpringMVC源码解析- HandlerAdapter - ModelFactory(转)

    ModelFactory主要是两个职责: 1. 初始化model 2. 处理器执行后将modle中相应参数设置到SessionAttributes中 我们来看看具体的处理逻辑(直接充当分析目录): 1 ...

  2. SpringMVC源码解析- HandlerAdapter - ModelFactory

    ModelFactory主要是两个职责: 1. 初始化model 2. 处理器执行后将modle中相应参数设置到SessionAttributes中 我们来看看具体的处理逻辑(直接充当分析目录): 1 ...

  3. springMVC源码解析--ViewResolver视图解析器执行(三)

    之前两篇博客springMVC源码分析--ViewResolver视图解析器(一)和springMVC源码解析--ViewResolverComposite视图解析器集合(二)中我们已经简单介绍了一些 ...

  4. springMVC源码解析--HandlerMethodArgumentResolverComposite参数解析器集合(二)

    上一篇博客springMVC源码分析--HandlerMethodArgumentResolver参数解析器(一)中我们已经介绍了参数解析相关的东西,并且也提到了HandlerMethodArgume ...

  5. SpringMVC源码解析- HandlerAdapter初始化

    HandlerAdapter初始化时,主要是进行注解解析器初始化注册;返回值处理类初始化;全局注解@ControllerAdvice内容读取并缓存. 目录: 注解解析器初始化注册:@ModelAttr ...

  6. SpringMVC源码解析

    一:springmvc运行过程: 1. dispatcherServlet 通过 HandlerMapping 找到controller2. controller经过后台逻辑处理得到结果集modela ...

  7. 深入了解SpringMVC源码解析

    Spring MVC源码解析 Spring MVC的使用原理其实是通过配置一个Servlet来接管所有的请求,所有的请求由这个Servlet来进行分发处理. 我们可以从web.xml里面看出这一点 & ...

  8. springMVC源码解析--ViewResolverComposite视图解析器集合(二)

    上一篇博客springMVC源码分析--ViewResolver视图解析器(一)中我们介绍了一些springMVC提供的很多视图解析器ViewResolver,在开发的一套springMVC系统中是可 ...

  9. springmvc源码解析MvcNamespaceHandler之<mvc:view-resolvers>

    说在前面 本次主要介绍springmvc配置解析. springmvc配置解析 本次介绍MvcNamespaceHandler. 进入到这个方法org.springframework.web.serv ...

随机推荐

  1. WebDriverException: Message: 'phantomjs.exe' executable needs to be in PATH.

    本文转载自:http://blog.csdn.net/sinat_36764186/article/details/55520444 网上的某测试代码: from selenium import we ...

  2. Java 判断是否包含指定的子串 contains()

    Java 手册 contains public boolean contains(CharSequence s) 当且仅当此字符串包含指定的 char 值序列时,返回 true. 参数: s - 要搜 ...

  3. Bootstrap-CL:分页

    ylbtech-Bootstrap-CL:分页 1.返回顶部 1. Bootstrap 分页 本章将讲解 Bootstrap 支持的分页特性.分页(Pagination),是一种无序列表,Bootst ...

  4. Linux常用系统函数

    Linux常用系统函数 一.进程控制 fork 创建一个新进程clone 按指定条件创建子进程execve 运行可执行文件exit 中止进程_exit 立即中止当前进程getdtablesize 进程 ...

  5. 【POJ】3378 Crazy Thairs(树状数组+dp+高精)

    题目 传送门:QWQ 分析 题意:给个数列,求有多少五元上升组 考虑简化一下问题:如果题目求二元上升组怎么做. 仿照一下逆序对,用树状数组维护一下就ok了. 三元怎么做呢? 把二元的拓展一位就可以了, ...

  6. 【UVA】11825 Hackers' Crackdown(状压dp)

    题目 传送门:QWQ 分析 $ n<= 16 $ 显然是状压 然后搞一搞(靠着蓝书yy一下) 代码 #include <bits/stdc++.h> using namespace ...

  7. 给iOS开发新手送点福利,简述UIControl事件的用法

    UIControl事件 1.UIControlEventTouchDown 单点触摸按下事件:用户点触屏幕,或者又有新手指落下的时候. 2.UIControlEventTouchDownRepeat ...

  8. C细节学习

    字符串ascii码值比较compress函数;

  9. leetcode877

    public class Solution { public bool StoneGame(int[] piles) { return true; } } 这问题很不好...

  10. eclipse Android 开发基础 Activity 窗体 界面

    eclipse Android 开发基础 新建工程 新建布局layout,new Android Activity就相当于窗体Form. 新建Activity自动生成src下同名的java代码. pu ...