从DispatcherServlet说起,本文讨论的内容都是DispatcherServlet的doDispatch方法完成

mappedHandler是一个HandlerExecutionChain,其中封装了一个handler,和一个由interceptor组成的list

如果这张图够高,往下能看到handler是如何执行的,不过不是handler.execute()这样的形式,而是由这样的:

// Actually invoke the handler.
mv = ha.handle(processedRequest, response, mappedHandler.getHandler());

从注释就可以看出,是通过反射的方式来调用,我已经看过一遍代码了,实际上里面是通过找到Method的对象,进行的反射调用,重点就在于,这个Method对象是如何找到的

而找的这个过程,是包含在第一张图中的下述方法调用里的:

HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());

这里使用的handlerAdapters是在private void initHandlerAdapters(ApplicationContext context)方法中得到初始化的,大致思想就是将所有实现了HandlerAdapter接口的类的实例都找出来放到这个list中

ha 是 AnnotationMethodHandlerAdapter 的实例时:执行上图的逻辑

getMethodResolver的逻辑,已经执行执行中methodResolverCache的内容如上图,可以看出里面是我们的handler对应的具体的Adapter对象

HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());

所以上面的ha实际上是比如:org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter@52d5661c

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

好吧,接着往下看,就是调用了,上面也提到过的,就是上面这段代码

它里面是上面这样,代码挺多,但我只关心return invokeHandlerMethod(request, response, handler);

protected ModelAndView invokeHandlerMethod(HttpServletRequest request, HttpServletResponse response, Object handler)
throws Exception { ServletHandlerMethodResolver methodResolver = getMethodResolver(handler);
Method handlerMethod = methodResolver.resolveHandlerMethod(request);
ServletHandlerMethodInvoker methodInvoker = new ServletHandlerMethodInvoker(methodResolver);
ServletWebRequest webRequest = new ServletWebRequest(request, response);
ExtendedModelMap implicitModel = new BindingAwareModelMap(); Object result = methodInvoker.invokeHandlerMethod(handlerMethod, handler, webRequest, implicitModel);
ModelAndView mav =
methodInvoker.getModelAndView(handlerMethod, handler.getClass(), result, implicitModel, webRequest);
methodInvoker.updateModelAttributes(handler, (mav != null ? mav.getModel() : null), implicitModel, webRequest);
return mav;
}

上面就是具体的调用。看到这里,整个过程都结束了,已经返回了ModelAndView

但是但是但是但是,很多个但是。只有认真看了,或者像我一样,尝试学习这样的实现,才会注意到很多细节。比如上面这个方法实际上只传了一个handler进来,它怎么知道invoke哪个Method呢???

ServletHandlerMethodResolver methodResolver = getMethodResolver(handler);
Method handlerMethod = methodResolver.resolveHandlerMethod(request);

重点就是上面这个methodResolver了,可以看到里面有一个mappings,存储了被@requestMapping注解的方法,和注解的值这样的一个对应关系

这是全文的重中之重

这个方法只有在服务器启动后,第一次接受请求时,才能走到resolver = new ServletHandlerMethodResolver(handlerClass);后面的都是使用methodResolverCache了

上面的init实现如下

这里面很关键,但是很不起眼的代码,竟然还被杠掉了,有空看看新的实现是什么,这段代码就是:if (isHandlerMethod(specificMethod)),

这个方法是属于AnnotationMethodHandlerAdapter类的,这个方法的参数就是Method,比如调试的时候是public java.lang.String com.test.springmvc.HelloWorld.hello()

上面这段代码执行的时候,

mapping的内容是:

@org.springframework.web.bind.annotation.RequestMapping(name=, value=[/helloworld], path=[/helloworld], method=[], params=[], headers=[], consumes=[], produces=[])

patterns的内容是

[/helloworld]

所以this.mappings.put(method, mappingInfo);的结果是

{public java.lang.String com.test.springmvc.HelloWorld.hello()=[/helloworld]}

就这样吧,明天来实现一把,当然不想按照它这个流程来,太复杂了,我最好简化一下

springbank 开发日志 SpringMVC是如何找到handler找到对应的方法并执行的的更多相关文章

  1. springbank 开发日志 阅读spring mvc的源代码真是受益良多

    决定模仿spring mvc的dispatcher->handlerMapping(return executorChain)->handler.execute 这样的流程之后,就开始看s ...

  2. springbank 开发日志 springbank是如何注册handler的

    这要从DefaultAnnotationHandlerMapping这个类说起,该类被@Component注释,该类被Spring IOC容器实例化之后,将会执行其initApplicationCon ...

  3. springbank 开发日志 springbank是如何执行一个handler的requestMapping对应的方法的

    占位 从dispatcher说起,方法doDispatch(Map request)的参数request是一个通过解析来报报文新城的map //获取HandlerExecutionChain,其中封装 ...

  4. springbank 开发日志 Spring启动过程中对自定义标签的处理

    这篇随笔的许多知识来源于:http://www.importnew.com/19391.html 之所以会去看这些东东,主要是希望能够模仿spring mvc的处理流程,做出一套合理的交易处理流程. ...

  5. SpringBank 开发日志 使用maven构建dubbo服务的可执行jar包

    写这篇日志的时候,我已经完成了这个目标,并且中间经历了一次面试.现在回过头看,已经觉得印象不那么深刻了,果然还是一边思考,一边记录这样最好.但我还是严格要求自己,从新做了梳理,对相关配置进行了整理和说 ...

  6. SpringBank 开发日志 Mybatis 使用redis 作为二级缓存时,无法通过cacheEnabled=false 将其关闭

    <?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE configuration PUBLIC ...

  7. SpringBank 开发日志 一种简单的拦截器设计实现

    当交易由Action进入Service之前,需要根据不同的Service实际负责业务的不同,真正执行Service的业务逻辑之前,做一些检查工作.这样的拦截器应该是基于配置的,与Service关联起来 ...

  8. SpringBank 开发日志 重新设计Action调用Service的参数传递 使用泛型解决类型转换问题

    之前想的比较简单,请求到达controller的时候,传给action的参数没有经过任何封装,就是一个Map.然后action再调用service的时候,传递的参数也是map @Controller ...

  9. springbank 开发日志 一次因为多线程问题导致的applicationContext.getBean()阻塞

    几天前遇到的这个问题.由于交易是配置的,不同的交易是同一个类的不同实例,所以不可能提前将其以@autowired类似的方式注入到需要的类中 <op:transaction id="Re ...

随机推荐

  1. Redis模块学习笔记(一)RediSearch简单使用

    说明:安装的Redis服务器必须为 4.0 以上版本,通过info命令查看 > INFO redis_version: 一.安装 RediSearch git clone https://git ...

  2. HTTP协议(下午茶)

    http://www.kancloud.cn/kancloud/tealeaf-http/43840   下午茶

  3. Tessaract 源码分析(转)

    源码分析 Page Layout 分析步骤 二值化算法: OTSU调用栈:main[api/tesseractmain.cpp] -> TessBaseAPI::ProcessPages[api ...

  4. js正则匹配table,img及去除各种标签问题

    //获取公示栏内容 s = "$row.detail$"; mainContent =s; //如果有多个table使用下面注释的正则只会匹配成一个table //var tabR ...

  5. JavaSE之Math类

    下列哪个选项是正确计算42度(角度)的余弦值? double d=Math.cos(42) double d=Math.cosine(42) double d=Math.cos(Math.toRadi ...

  6. [转]NOI_Linux Arbiter使用手册

    讲述清楚,简单易懂的Arbiter使用手册 转载自 https://www.cnblogs.com/gengchen/p/7761565.html Arbiter 系统使用说明 Overview Ar ...

  7. JS 中对变量类型判断的几种方式

    文章整理搬运,出处不详,如有侵犯,请联系~   数据类型判断和数据类型转换代码工具 在 JS 中,有 5 种基本数据类型和 1 种复杂数据类型,基本数据类型有:Undefined, Null, Boo ...

  8. webstorm11.0.3连接ftp

    一.工具 webstorm11.0.3 ftp工具:Beyond Compare 3 二.步骤 1.打开项目工程,按照下图中路径,打开ftp配置界面 2.在ftp配置界面 (1)右上角+号新建配置,选 ...

  9. mysql 架构 ~异地容灾

    一 简介 我们来探讨下多机房下的mysql架构二 目的:    首先要清楚你的目的     1 实现异地机房的容灾备份      2 实现异地机房的双活 三 叙说     1 实现异地机房的容灾备份  ...

  10. 使用sqlmap中tamper脚本绕过waf

    使用sqlmap中tamper脚本绕过waf 刘海哥 · 2015/02/02 11:26 0x00 背景 sqlmap中的tamper脚本来对目标进行更高效的攻击. 由于乌云知识库少了sqlmap- ...