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. 每天一个 HTTP 状态码 201

    201 Created 201 Created 表示客户端的请求已经成功完成,结果是创建了一个新资源,通常用于响应「增删改查」里的「增」.如果是严格按照 RESEful style 的 API,那么当 ...

  2. Vue2-Slot插槽使用

    Slot插槽 父组件向子组件传递 父组件将内容分发到子组件 slot插槽的值只读,不能在子组件中修改 slot插槽也可以作为组件之间的通信方式 默认插槽 父组件中:使用Son组件 <templa ...

  3. Vben Admin 源码学习:项目初始化

    0x00 前言 Vue-Vben-Admin 是一个免费开源的中后台模版.使用了最新的vue3,vite2,TypeScript等主流技术开发,开箱即用的中后台前端解决方案考. 本系列本着学习参考的目 ...

  4. python自动将新生成的报告作为附件发送并进行封装

    发送报告作为自动化部署来讲是一个重要的环节,废话不多说直接上代码吧,如果想更细致的了解内容查阅本博主上篇基本发送文章 特别叮嘱一下:SMTP协议默认端口25,qq邮箱SMTP服务器端口是465 别出丑 ...

  5. 入坑KeePass(二)重置keepass设置

    保留好.kdbx和密钥文件,软件的文件可以删除掉,重新下载并解压设置就恢复默认了

  6. 29.MySQL高级SQL语句

    MySQL高级SQL语句 目录 MySQL高级SQL语句 创建两个表 SELECT DISTINCT WHERE AND OR IN BETWEEN 通配符 LIKE ORDER BY 函数 数学函数 ...

  7. JS:函数

    Function:函数 1. 定义一个函数:function functionname(argument) { 代码块 return }: 调用此函数:fn() 2.函数是定义了一种方法,只有被调用才 ...

  8. JavaScript产生随机颜色

    //获取rgb类型的颜色 IE7不支持 function randomColor(){ var r = Math.floor(Math.random()*256); var g = Math.floo ...

  9. 全新升级的AOP框架Dora.Interception[6]: 框架设计和实现原理

    本系列前面的五篇文章主要介绍Dora.Interception(github地址,觉得不错不妨给一颗星)的编程模式以及对它的扩展定制,现在我们来聊聊它的设计和实现原理.(拙著<ASP.NET C ...

  10. 端口被占用的问题解决 Web server failed to start. Port ×× was already in use

    出现此问题是端口被占用了,只需要关闭正在使用的端口就行 解决思路: 1.在服务器中更改port端口号,改为不冲突,没有被占用的端口. 2.找出被占用的端口,结束被占用的端口 解决结束被占用的端口的方法 ...