Spring mvc之源码 handlerMapping和handlerAdapter分析

本篇并不是具体分析Spring mvc,所以好多细节都是一笔带过,主要是带大家梳理一下整个Spring mvc的执行流程,以及如何根据URL查找处理器Controller的实现

(适合那些刚阅读源码不知道如何下手的人)

http://www.guojinbao.com/borrow/borrowDetail/GETadLPjnf0[d].do

如何根据URL地址----》找到正确处理器Controller的相应方法Method

注:概念 IOC容器:负责bean的创建和维护bean与bean关系的一个大的集合

①dispatcherServlet 在Spring MVC为所有请求入口,标准的servlet。

标准的servlet执行流程为

1.init()  首次加载servlet 整个servlet生命周期只执行一次

2.service()  依据url-pattern(<url-pattern>/</url-pattern> 拦截所有请求,不包括静态资源文件) 拦截模式,匹配请求模式,匹配成功调用service方法

3.destroy()  servlet容器(如:tomcat停止)结束调用

执行流程如下

第一步 执行 HttpServletBean-->init()方法

init()方法我们只需要关注initServletBean()方法 此方法为抽象方法由FrameworkServlet实现

有兴趣的可以具体看一下源码,整个init()方法 为spring mvc的IOC容器初始化

注:Spring 的IOC容器初始化由contextLoaderListener完成

Spring IOC容器建议配置:数据库连接,事务,与第三方框架集成等等

如:ApplicationContext.xml  (小提示:处理器Controller的扫描 可以交给springMVC IOC容器完成,目的分层明确,便于IOC容器查找 )  

Spring mvc IOC 容器 配置:

第二部 拦截请求

① FrameworkServlet -->service()

分发请求

② FrameworkServlet--->processRequest(request, response);

③ DispatcherServlet--->doService(HttpServletRequest request, HttpServletResponse response)

如何根据URL找到处理器执行正确的方法就在此方法中

④ DispatcherServlet--->doDispatch(HttpServletRequest request, HttpServletResponse response)

第①部分

这里为啥不是调用super.service(request,response)?

最直接的理由:如果直接调用service方法 根据method类型进行分发请求就直接跳出框架啦(你懂的嘿嘿)

②③2步有兴趣的可以看一下源码

第④部分

DispatcherServlet--->doDispatch(HttpServletRequest request, HttpServletResponse response)

整个过程如下:

>>1. 获取url对应的处理器controller以及相应方法

>>2. 获取执行处理器中方法的对象

>>3. 具体执行

Spring mvc是如何通过URL查找到对应的处理器controller以及对应的方法尼?需要知道以下3个核心接口handlerMapping,handlerAdapter,RequestCondition我们不具体分析相应接口的实现以及具体逻辑,过程相当复杂,1到2句话说

不清楚,这里给大家留一个直观的考量。

由于Spring mvc源码庞大,建议看源码的时候,只看重点部分或者说自己感兴趣部分。我们大家常用是在类和方法加上注解@RequestMapping来标识访问路径,我们以此模板给大家量化分析url查询处理器controller以及相应方法的过程

handlerMapping:负责查找URL对应的处理器Controller

RequestMappingInfo:根据注解@RequestMapping生成方法匹配策略

handlerAdapter:负责绑定参数,执行请求,处理返回值

handlerMapping 接口就一个方法返回HandlerExecutionChain

HandlerExecutionChain

DispatcherServlet--->doDispatch()

第一步 获取HandlerExecutionChain

在DispatcherServlte的servlet初始化中(调用init()方法)中调用onRefresh()完成handlerMappings的初始化

第二步 获取handlerAdapter 找到和handlerMapping匹配的handlerAdapter

初始化过程同第一步

DispatcherServlte会根据handlerMapping与已经注册好了的HandlerAdapter一一匹配,看哪一种HandlerAdapter是支持该handlerMapping类型的,

如果找到了其中一种HandlerAdapter是支持传过来的handlerMapping类型,那么该HandlerAdapter会调用自己的handle方法,handle方法运用Java的反射机制执行controller的具体方法来获得ModelAndView

以handlerMapping的实现RequestMappingHandlerMapping为例

由于实现了InitializingBean,当RequestMappingHandlerMapping完成bean的初始化后,会调用afterPropertiesSet()-->initHandlerMethods();完成以下操作

①生成直接URL-->RequestMappingInfo,映射操作

②生成 RequestMappingInfo--->handlerMethod映射操作

③将所有的RequestMappingInfo注册到容器中

注:

RequestMappingInfo实现了RequestCondition接口 存储着根据注解@RequestMapping生成方法匹配策略

举个例子

PatternsRequestCondition的初始化相当于

PatternsRequestCondition(“/user/applicationShow/{id}”)当请求URL符合此规则是就匹配成功

简单滴说,RequestMappingInfo是将注解@RequestMapping的信息提取出来

生成此方法的匹配条件

阅读RequestMappingHandlerMapping源码流程如下(主要是为了梳理出重点部分)

综上所述我们来猜测一下spring mvc 中根据URL找到处理器Controller中相应方法的流程

①:获取Request的URL

②:从UrlLookup这个map中找到相应的requestMappingInfo

③:如果没找到则遍历所有requestMappingInfo的信息直到找到匹配的requestMappingInfo

④:根据requestMappingInfo到mappingLookup中找到handlerMethod

handlerMethod接口如下:

到这里大家差不多就明白了吧,有了处理器实例,方法,参数调用反射执行方法分分钟的事情

当然spring mvc真正执行比这个复杂的多啦,但是执行的时候是逃不出这个套路的。

具体的URL找到处理器Controller中相应方法的流程请大家看一下DispatcherServlet--》getHandler()方法

注:

最后说一下RequestMappingHandlerMapping并不是默认的查找处理器方法的策略,他是通过  <mvc:annotation-driven />配置完成容器注册的

Spring mvc之源码 handlerMapping和handlerAdapter分析的更多相关文章

  1. Spring MVC之源码速读之RequestMappingHandlerAdapter

    spring-webmvc-4.3.19.RELEASE 下面来看DispatcherServlet中的执行: /** * Exposes the DispatcherServlet-specific ...

  2. Spring IOC 容器源码分析 - 余下的初始化工作

    1. 简介 本篇文章是"Spring IOC 容器源码分析"系列文章的最后一篇文章,本篇文章所分析的对象是 initializeBean 方法,该方法用于对已完成属性填充的 bea ...

  3. Spring IOC 容器源码分析 - 填充属性到 bean 原始对象

    1. 简介 本篇文章,我们来一起了解一下 Spring 是如何将配置文件中的属性值填充到 bean 对象中的.我在前面几篇文章中介绍过 Spring 创建 bean 的流程,即 Spring 先通过反 ...

  4. Spring IOC 容器源码分析 - 循环依赖的解决办法

    1. 简介 本文,我们来看一下 Spring 是如何解决循环依赖问题的.在本篇文章中,我会首先向大家介绍一下什么是循环依赖.然后,进入源码分析阶段.为了更好的说明 Spring 解决循环依赖的办法,我 ...

  5. Spring IOC 容器源码分析 - 创建原始 bean 对象

    1. 简介 本篇文章是上一篇文章(创建单例 bean 的过程)的延续.在上一篇文章中,我们从战略层面上领略了doCreateBean方法的全过程.本篇文章,我们就从战术的层面上,详细分析doCreat ...

  6. Spring IOC 容器源码分析 - 创建单例 bean 的过程

    1. 简介 在上一篇文章中,我比较详细的分析了获取 bean 的方法,也就是getBean(String)的实现逻辑.对于已实例化好的单例 bean,getBean(String) 方法并不会再一次去 ...

  7. Spring IOC 容器源码分析 - 获取单例 bean

    1. 简介 为了写 Spring IOC 容器源码分析系列的文章,我特地写了一篇 Spring IOC 容器的导读文章.在导读一文中,我介绍了 Spring 的一些特性以及阅读 Spring 源码的一 ...

  8. Spring IOC 容器源码分析系列文章导读

    1. 简介 Spring 是一个轻量级的企业级应用开发框架,于 2004 年由 Rod Johnson 发布了 1.0 版本.经过十几年的迭代,现在的 Spring 框架已经非常成熟了.Spring ...

  9. Spring IOC 容器源码分析

    声明!非原创,本文出处 Spring 最重要的概念是 IOC 和 AOP,本篇文章其实就是要带领大家来分析下 Spring 的 IOC 容器.既然大家平时都要用到 Spring,怎么可以不好好了解 S ...

随机推荐

  1. redis等缓存

    文章出处 https://www.cnblogs.com/wupeiqi/articles/5246483.html Model 到目前为止,当我们的程序涉及到数据库相关操作时,我们一般都会这么搞: ...

  2. 网络游戏开发-客户端2(自定义websocket协议格式)

    Egret官方提供了一个Websocket的库,可以让我们方便的和服务器长连接交互. 标题写的时候自定义websocket的协议格式.解释一下,不是说我们去动websocket本身的东西,我们是在we ...

  3. 网页布局——float浮动布局

    我的主要参考资料是[Object object]的文章 float 布局应该是目前各大网站用的最多的一种布局方式了,但是也特别复杂,这里详细讲一下 首先,什么是浮动? 浮动元素是脱离文档流的,但不脱离 ...

  4. PhpSpreadsheet 导出特定格式 — 广告请款单

    需求说明 最近需要实现一个导出这种格式的Excel表单,之前都有用过导出Excel的功能,但大都是表头+数据的形式,只用于获取数据,没有太多样式要求,不用合并单元格.合并居中等,也不用对每一行数据特异 ...

  5. 拿起键盘就是干:跟我一起徒手开发一套分布式IM系统

    1.引言 老读者应该还记得我在去年国庆节前分享过一篇<技术干货:从零开始,教你设计一个百万级的消息推送系统>,虽然我在文中有贴一些伪代码,依然有些朋友希望能直接分享一些可以运行的源码.好吧 ...

  6. 三种常见字符编码:ASCII、Unicode和UTF-8

    什么是字符编码? 计算机只能处理数字,如果要处理文本,就必须先把文本转换为数字才能处理.最早的计算机在设计时采用8个比特(bit)作为一个字节(byte),所以,一个字节能表示的最大的整数就是255( ...

  7. POJ 1753 Flip Game(状态压缩+BFS)

    题目网址:http://poj.org/problem?id=1753 题目: Flip Game Description Flip game is played on a rectangular 4 ...

  8. 去除提示“Chrome正在受到自动软件的控制”

    现象:用selenium启动浏览器出现‘Chrome正在受到自动软件的控制’ 解决办法:在启动浏览器前加上如下配置 # 加启动配置 option = webdriver.ChromeOptions() ...

  9. ubuntu 虚拟机设置静态ip

    $ sudo vim /etc/network/interfaces auto ens33   # 使用的网络接口,之前查询接口是为了这里     iface ens33 inet static    ...

  10. liunux中的标准输出。以及常用的 2>dev/null 命令的含义

    了解Linux怎样处理输入和输出是非常重要的.一旦我们了解其原理以后,我们就可以正确熟练地使用脚本把内容输出到正确的位置.同样我们也可以更好地理解输入重定向和输出重定向. 首先我们来了解一下linux ...