在DispatcherServlet.java的doDispatch方法中,springmvc通过handlermapping里面找哪个handler能处理请求,handler封装了目标方法的信息,

mappedHandler = getHandler(processedRequest);

然后为当前的handler找到一个适配器HandlerAdapter,寻找的过程为:在DispatcherServlet.java的getHandlerAdapter方法中,挨个匹配,判断当前adapter是否支持当前handler,判断方法为只要handler是handlerMethod类型就生效,就支持

HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());
if (this.handlerAdapters != null) {
for (HandlerAdapter adapter : this.handlerAdapters) {
if (adapter.supports(handler)) {
return adapter;
}
}
}

找到适配器以后判断当前请求是不是“GET”方法以及“HEAD”,“HEAD”不是服务器真正处理的

适配器HandlerAdapter把(目标方法、request、response)传入handle执行目标方法

mv = ha.handle(processedRequest, response, mappedHandler.getHandler());

怎么执行目标方法:

  • 先得到handler
return this.handler;
  • 再进入内部处理细节RequestMappingHandlerAdapter.java,调用的invokeHandlerMethod就是执行目标方法
mav = invokeHandlerMethod(request,response,handlerMethod);
  • 在RequestMappingHandlerAdapter.java的invokeHandlerMethod方法中,

    • 为invocableMethod方法设置参数解析器argumentResolvers,参数解析器确定将要执行的目标方法的每一个参数的值是什么

      • 当前解析器是否支持解析这种参数

      • 支持就调用 resolveArgument

    • 为invocableMethod方法设置返回值处理器returnValueHandlers

  • 把26个argumentResolvers和15个returnValueHandlers都放入目标方法包装的ServletInvocableHandlerMethod中

  • 然后真正执行目标方法的语句

    invocableMethod.invokeAndHandle(WebRequest,mavContainer);
    • 在ServletInvocableHandlerMethod.java的invokeAndHandle方法中,执行了controller

      Object returnValue = invokeForRequest(webRequest,mavContainer,proviedArgs);
      • step into 进入InvocableHandlerMethod.java,确定目标方法每一个参数的值

        Object[] agrs = getMethodArgumentValues(request,mavContainer,providerArgs)
        • 在InvocableHandlerMethod.java的getMethodArgumentValues方法中,先获取方法所有的参数声明(详细信息)。

          MethodParameter[] parameters = getMethodParameters();
        • 判断参数是否为空,为空则无需确定任何值直接返回;

          if (ObjectUtils.isEmpty(parameters)) {
          return EMPTY_ARGS;
          }

          如果有参数列表,new一个Object[],参数列表有多少个Object[]就有多长

          Object[] args = new Object[parameters.length];
          for (int i = 0; i < parameters.length; i++) {
          MethodParameter parameter = parameters[i];
          parameter.initParameterNameDiscovery(this.parameterNameDiscoverer);
          args[i] = findProvidedArgument(parameter, providedArgs);

          先声明args遍历parameters,给args[i]赋值,args[i]的值解析器解析了才有

        • 解析之前,判断26个解析器是不是supportsParamter支持这个参数类型。

          this.resolvers.supportsParameter(parameter)
          • HandlerMethodArgumentResolverComposite.java的getArgumentResolver方法中

          • 获取一个缓存result

          • result==null,进入增强for循环,逐个确定26个解析器谁能支持这种参数

            • supportsParameter方法,传来的参数有没有hasParameterAnnotation标注注解
            • 没有就return false
            • 如果标了,再判断参数是否map类型,
            • return true支持解析
          • 当前resolver支持解析,放到缓存里边,判断成功,进入解析

        • 解析参数

          this.resolvers.resolveArgument
          • HandlerMethodArgumentResolverComposite.java的resolverArgument方法中,先拿到所有的参数解析器getArgumentResolver
          • 调用参数解析器的resolverArgument方法进行解析
            • 获取参数名字信息
            • 解析参数的名字,placeholderResolved、BeanExpressionResolver解析evaluate计算名字,按照正则匹配的方式
            • 解析参数的值
              • uriTemlateVars 在request请求域中拿到值;UrlPathHelper会把uri地址里边的所有的路径变量全部解析出来并保存到请求域中
        • 遍历循环所有参数

      • 最终返回args,args就是确定好的值

    • 处理返回结果的时候,把mavContainer传进去,

      this.returnValueHandlers.handlerReturnValue
      • 在handlerMethod.java的getReturnValueType方法中获取返回的结果类型

      • HandlerMethodArgumentResolverComposite.java的handleReturnValue方法中,

        找到返回值的处理器

        如果返回值是一个字符串,拿到字符串然后保存到mavContainer

    • 返回值处理完以后,getModelAndView

  • 目标方法执行完成

    将所有的数据都放在 ModelAndViewContainer;包含要去的页面地址View。还包含Model数据。

  • 从ModelAndViewContainer拿到默认的Model,updateBindingResult,拿到key放到绑定里边,又被封装成ModelAndView,然后返回这个新封装的mav

    • 处理派发结果

      processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException);

      拿到所有请求域中的属性,解析得到视图名

      渲染页面

      view.render(mv.getModelInternal(),request,response);

      拿到页面数据

      createMergeOutputModel(model,request,response);
      创造合并的输出模型

      如果model不等于空

      mergeModel.putAll(model)//即把数据转移到HashMap

      渲染合并输出的模型数据

      renderMergedOutputModel(mergedModel, getRequestToExpose(request), response);
      • 把上面的Hashmap传进来了

      • 拿到请求对象,获取的原生的Servletrequest

        return originalRequest
      • 暴露模型作为请求域属性

        // Expose the model object as request attributes.
        exposeModelAsRequestAttributes(model, request);
        • 把model里面的东西进行遍历
        • 遍历以后每一个request setAttribute

SpringMVC底层——请求参数处理流程描述的更多相关文章

  1. 16 SpringMVC 的请求参数的绑定与常用注解

    1.SpringMVC 绑定请求参数 (1)支持的数据类型 基本类型参数: 包括基本类型和 String 类型POJO 类型参数: 包括实体类,以及关联的实体类数组和集合类型参数: 包括 List 结 ...

  2. SpringMVC之请求参数的获取方式

    转载出处:https://www.toutiao.com/i6510822190219264516/ SpringMVC之请求参数的获取方式 常见的一个web服务,如何获取请求参数? 一般最常见的请求 ...

  3. SpringMVC处理请求和返回流程

    流程描述:一个url请求,找打指定的requestMapping再返回指定的jsp界面 通过url拿到指定的java方法 HandlerExecutionChain  mappedHandler = ...

  4. SpringMVC接受请求参数、

    1. 接收请求参数 1.1. [不推荐]通过HttpServletRequest 在处理请求的方法中,添加HttpServletRequest对象作为参数,在方法体中,直接调用参数对象的getPara ...

  5. SpringMVC RequestMapping & 请求参数

    SpringMVC 概述 Spring 为展现层提供的基于 MVC 设计理念的优秀的Web 框架,是目前最主流的 MVC 框架之一 Spring3.0 后全面超越 Struts2,成为最优秀的 MVC ...

  6. SpringMVC 接受请求参数、作用域传值

    目录 原生servlet接收参数 Spring MVC最基础的参数获取 接收基本数据类型参数 方法参数列表和请求参数不一致的处理方式 接收对象引用数据类型 接收复选框这种多个同名的参数 接收obj.f ...

  7. SpringMVC 获取请求参数

    1.获取Request response对象 在SpringMVC的注解开发中,可以选择性的接收Request和Response对象来使用 2.获取request对象请求参数 a.通过request对 ...

  8. SpringMVC——映射请求参数

    Spring MVC 通过分析处理方法的签名,将 HTTP 请求信息绑定到处理方法的相应人参中. @PathVariable @RequestParam @RequestHeader 等) Sprin ...

  9. SpringMVC获取请求参数-POJO类型参数

    1.Controller中的业务方法的POJO参数的属性名与请求参数一致,参数值会自动映射匹配 1.创建POJO类 public class User { private String usernam ...

随机推荐

  1. .NET性能优化-为结构体数组使用StructLinq

    前言 本系列的主要目的是告诉大家在遇到性能问题时,有哪些方案可以去优化:并不是要求大家一开始就使用这些方案来提升性能. 在之前几篇文章中,有很多网友就有一些非此即彼的观念,在实际中,处处都是开发效率和 ...

  2. 印尼医疗龙头企业Halodoc的数据平台转型之Lakehouse架构

    1. 摘要 在 Halodoc,我们始终致力于为最终用户简化医疗保健服务,随着公司的发展,我们不断构建和提供新功能. 我们两年前建立的可能无法支持我们今天管理的数据量,以解决我们决定改进数据平台架构的 ...

  3. 检查MySQL主从复制运行状态

    脚本思路: 1.使用MySQL客户端命令"mysql"直接在命令行中运行MySQL指令"show slave status\G;"来查看MySQL主从复制状态信 ...

  4. vue上传图片的3种方式

    https://blog.csdn.net/q3254421/article/details/88250968?utm_medium=distribute.pc_relevant.none-task- ...

  5. 03-数据结构(C语言版)

    Day01 笔记 1 数据结构基本理论 1.1 算法五个特性: 1.1.1 输入.输出.有穷.确定.可行 1.2 数据结构分类 1.2.1 逻辑结构:集合.线性.树形.图形 1.2.2 物理结构:顺序 ...

  6. SSE图像算法优化系列三十二:Zhang\Guo图像细化算法的C语言以及SIMD指令优化

    二值图像的细化算法也有很多种,比较有名的比如Hilditch细化.Rosenfeld细化.基于索引表的细化.还有Opencv自带的THINNING_ZHANGSUEN.THINNING_GUOHALL ...

  7. 软件构造Lab2实验总结

    本次实验训练抽象数据类型(ADT)的设计.规约.测试,并使用面向对象编程(OOP)技术实现ADT.具体来说内容如下: 针对给定的应用问题,从问题描述中识别所需的ADT: 设计ADT规约(pre-con ...

  8. AtCoder ABC 250 总结

    AtCoder ABC 250 总结 总体 连续若干次一样的结果:30min 切前 4 题,剩下卡在 T5 这几次卡在 T5 都是一次比一次接近, 什么 dp 前缀和打挂,精度被卡,能水过的题连水法都 ...

  9. Camunda定时器事件示例Demo(Timer Events)

    ​Camunda定时器事件(Timer Events)是由定义的计时器触发的事件.它们可以用作启动事件.中间事件或边界事件.边界事件可以中断,也可以不中断. Camunda定时器事件包括:Timer ...

  10. 【转载】k8s入坑之路(2)kubernetes架构详解

    每个微服务通过 Docker 进行发布,随着业务的发展,系统中遍布着各种各样的容器.于是,容器的资源调度,部署运行,扩容缩容就是我们要面临的问题. 基于 Kubernetes 作为容器集群的管理平台被 ...