Model 和 Map

为什么在Model和Map中放值传入后会出现在request的上面。

9.1、源码解析

  1. 准备测试代码

    @GetMapping("/goto")
    public String go(HttpServletRequest request,
    Map<String,Object> map,
    Model model){ request.setAttribute("msg","传过来....");
    map.put("map","map hello word!!");
    model.addAttribute("model","model hello word!!"); return "forward:success";
    } @ResponseBody
    @GetMapping("/success")
    public Map success(@RequestAttribute(value = "msg",required = false) String msg,
    HttpServletRequest request){
    Map<String,Object> hashMap = new HashMap<>(); Object msg1 = request.getAttribute("msg");
    Object map = request.getAttribute("map");
    Object model = request.getAttribute("model"); hashMap.put("map1",map);
    hashMap.put("model",model);
    hashMap.put("msg1",msg1); return hashMap;
    }
  2. 第一步进入DispatchServlet 的 doDispatch中

    • 主要的三步

      DispatchServlet类中的
      doDispatch方法
      // Determine handler for the current request.
      mappedHandler = getHandler(processedRequest);//获取反射的方法处理器
      // Determine handler adapter for the current request.
      //获取参数处理器
      HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());
      // Actually invoke the handler.
      //反射执行方法和解析Model和Map
      mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
  3. 在获取参数阶段,解Model和Map

    • 获取Map 的 处理器器 MapMethodProcessor

    • 处理Map这个参数

      MapMethodProcessor类中的
      @Override
      @Nullable
      public Object resolveArgument(MethodParameter parameter,
      @Nullable ModelAndViewContainer mavContainer,
      NativeWebRequest webRequest,
      @Nullable WebDataBinderFactory binderFactory) throws Exception {
      Assert.state(mavContainer != null, "ModelAndViewContainer is required for model exposure"); return mavContainer.getModel();
      } 然后进入ModelAndViewContainer类中的
      private final ModelMap defaultModel = new BindingAwareModelMap();
      public ModelMap getModel() {
      if (useDefaultModel()) {
      return this.defaultModel;//获取ModelMap
      }
      ......
    • 把这个ModelMap对象返回

    • 获取Model的 处理器器 ModelMethodProcessor

    • 处理Model这个参数

      ModelMethodProcessor类中的
      @Override
      @Nullable
      public Object resolveArgument(MethodParameter parameter,
      @Nullable ModelAndViewContainer mavContainer,
      NativeWebRequest webRequest,
      @Nullable WebDataBinderFactory binderFactory) throws Exception { Assert.state(mavContainer != null, "ModelAndViewContainer is required for model exposure");
      return mavContainer.getModel();
      } 然后进入ModelAndViewContainer类中的
      private final ModelMap defaultModel = new BindingAwareModelMap();
      public ModelMap getModel() {
      if (useDefaultModel()) {
      return this.defaultModel;//获取ModelMap 这个和那个Map获取的是同一个
      }
      ......
    • 把同一个ModelMap对象返回

  4. 反射调用方法

    由于是同一个对象所有 map 和 model中的值都样,所以往map和Model放值都一样

  5. 把model和map的数据放入request

    • 第一步
    ServletInvocableHandlerMethod类中
    invokeAndHandle方法
    //获取返回值的一个处理器
    try {
    this.returnValueHandlers.handleReturnValue(
    returnValue, getReturnValueType(returnValue),
    mavContainer, webRequest);
    }
    HandlerMethodReturnValueHandlerComposite类中的
    HandlerMethodReturnValueHandler方法 进行获取

    这里获取的ViewNameMethodReturnValueHandler这个处理器

    • 第二步

      把Map和 Model携带的参数和返回值(路径)进行整合为ModelAndView

    RequestMappingHandlerAdapter类中
    handleInternal方法
    ...
    ModelAndView mav;
    ...
    mav = invokeHandlerMethod(request, response, handlerMethod);
    ...

    • 第三步

      准备派发和放入参数

      DispatchServlet类中的
      doDispatch方法
      render(mv, request, response);
      view.render(mv.getModelInternal(), request, response); AbstractView类中的
      render方法
      //把BindingAwareModelMap的参数封装为Map
      Map<String, Object> mergedModel = createMergedOutputModel(model,
      request, response);
      prepareResponse(request, response);
      //准备方法和放入参数到reques中
      renderMergedOutputModel(mergedModel, getRequestToExpose(request),
      response);
      InternalResourceView类中的
      renderMergedOutputModel
      // Expose the model object as request attributes.
      exposeModelAsRequestAttributes(model, request);//把参数放入到request中
      AbstractView类中的
      //把参数放入到原生的request域中
      protected void exposeModelAsRequestAttributes(Map<String, Object> model,
      HttpServletRequest request) throws Exception {
      model.forEach((name, value) -> {
      if (value != null) {
      request.setAttribute(name, value);
      }
      else {
      request.removeAttribute(name);
      }
      });
      }

Spring的Model 和 Map的原理的更多相关文章

  1. Spring框架系列(14) - SpringMVC实现原理之DispatcherServlet处理请求的过程

    前文我们有了IOC的源码基础以及SpringMVC的基础,我们便可以进一步深入理解SpringMVC主要实现原理,包含DispatcherServlet的初始化过程和DispatcherServlet ...

  2. 理解Spring MVC Model Attribute和Session Attribute

    作为一名 Java Web 应用开发者,你已经快速学习了 request(HttpServletRequest)和 session(HttpSession)作用域.在设计和构建 Java Web 应用 ...

  3. SpringMVC(二)--处理数据模型、ModelAndView、Model、Map、重定向、@ModelAttribute、

    1.处理模型数据 Spring MVC 提供了以下几种途径输出模型数据:      – ModelAndView:处理方法返回值类型为 ModelAndView 时, 方法体即可通过该对象添加模型数据 ...

  4. Spring中Model、ModelMap及ModelAndView之间的区别

    Spring中Model.ModelMap及ModelAndView之间的区别   1. Model(org.springframework.ui.Model)Model是一个接口,包含addAttr ...

  5. HashMap,Hashtable,ConcurrentHashMap 和 synchronized Map 的原理和区别

    HashMap 是否是线程安全的,如何在线程安全的前提下使用 HashMap,其实也就是HashMap,Hashtable,ConcurrentHashMap 和 synchronized Map 的 ...

  6. 【译】理解Spring MVC Model Attribute 和 Session Attribute

    作为一名 Java Web 应用开发者,你已经快速学习了 request(HttpServletRequest)和 session(HttpSession)作用域.在设计和构建 Java Web 应用 ...

  7. ModelAndView存取数Model、Map、ModelMap

    1.从名字上看ModelAndView中的Model代表模型,View代表视图,这个名字就很好地解释了该类的作用.(对应mvc配置,寻找模型,返回视图.) 2.业务处理器调用模型层处理完用户请求后,把 ...

  8. Spring Boot 实现配置文件加解密原理

    Spring Boot 配置文件加解密原理就这么简单 背景 接上文<失踪人口回归,mybatis-plus 3.3.2 发布>[1] ,提供了一个非常实用的功能 「数据安全保护」 功能,不 ...

  9. Spring框架系列(13) - SpringMVC实现原理之DispatcherServlet的初始化过程

    前文我们有了IOC的源码基础以及SpringMVC的基础,我们便可以进一步深入理解SpringMVC主要实现原理,包含DispatcherServlet的初始化过程和DispatcherServlet ...

随机推荐

  1. 【FineBI】增量数据更新语句

    SELECT * FROM t_abike_user WHERE AddUserTime BETWEEN '2016-11-17 10:49:04' AND '2021-07-31 23:59:59'

  2. 探究MySQL中SQL查询的成本

    成本 什么是成本,即SQL进行查询的花费的时间成本,包含IO成本和CPU成本. IO成本:即将数据页从硬盘中读取到内存中的读取时间成本.通常1页就是1.0的成本. CPU成本:即是读取和检测是否满足条 ...

  3. 关于我学git这档子事

    创建本地分支并切换到该分支 git checkout -b *** 相当于如下2个命令: git branch *** git checkout *** 推送本地开发分支到远程开发分支 git pus ...

  4. 背包,子集和以及 (max, +) 卷积在特殊情形下的求法

    背包,子集和以及 (max, +) 卷积在特殊情形下的求法 子集和 1:总重量不太大 有 \(n\) 个物品,每个物品重量为 \(w_i\),且 \(\sum\limits_{i} w_i=C\).你 ...

  5. java中的final与可变类型、不可变类型的关系

    如果你对final和不可变类型的概念与区别有疑问的话,可以打开这篇文章.希望我的解答可以帮到您! 1.不可变类型: 什么是可变类型,什么是不可变类型呢? 首先我们看一下下面的这行代码: String ...

  6. RocketMQ的基本使用

    第一步导入依赖: <!--Springboot 集成 RocketMQ依赖--> <dependency> <groupId>org.apache.rocketmq ...

  7. 如何写一个全局的 Notice 组件?

    下面将会实现这样的效果: 组件动态创建脚本: NotificationBanner.js import Vue from "vue"; import Notice from &qu ...

  8. 001 手把手用Git,Git从入门到上传本地项目到Github,看这篇就够了

    安装git 下载Git 下载好后,一路next即可 安装好后,打开Git bash,进行配置 首先配置自己的身份 git config --global user.name "Name&qu ...

  9. 开源流程引擎该如何选择flowable还是camunda

    市场上比较有名的开源流程引擎有osworkflow.jbpm.activiti.flowable.camunda.现在国内用的最多的是activiti.flowable.camunda,下面主要从功能 ...

  10. CVE-2022-30190 Follina Office RCE分析【附自定义word钓鱼模板POC】

    昨天看了下'Follina' MS-MSDT n-day Microsoft Office RCE 这个漏洞,修改了下chvancooten的脚本,实现可以自定义word模板,便于实战中钓鱼使用,自己 ...