从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. UESTC - 1324 卿学姐与公主

    题目链接 某日,百无聊赖的卿学姐打开了某11区的某魔幻游戏 在这个魔幻的游戏里,生活着一个美丽的公主,但现在公主被关押在了魔王的城堡中. 英勇的卿学姐拔出利刃冲向了拯救公主的道路. 走过了荒野,翻越了 ...

  2. Jupyter 魔术命令(magic commands)

    自动重新加载更改的模块 命令参数如下所示: %autoreload: 自动重载%aimport排除的模块之外的所有模块. %autoreload 0: 禁用自动重载 %autoreload 1: 自动 ...

  3. jsp/servlet页面跳转丢失样式问题

    问题:使用servlet,如何处理在多路径页面跳转中servlet转发页面样式丢失问题?(例如访问 http://localhost/project/listUser.action后转到http:// ...

  4. Mybatis中的StatementType

    原文:http://luoyu-ds.iteye.com/blog/1517607 要实现动态传入表名.列名,需要做如下修改 添加属性statementType=”STATEMENT” 同时sql里的 ...

  5. dijkstra补充

    dijkstra主要写法: priority_queue<pair<int,int> >q; //大根堆 //dis第一维为dis的相反数 void dijkstra(){ m ...

  6. C++使用目录

    VS2017的安装和配置 常用指令 C++ 数据类型   常量 运算符 数组 字符串  Ansi与Unicode  指针   模态与非模态对话框  变量的引用& new和delete动态分配和 ...

  7. greenplum不能下载问题解决方法(转)

    到官网下载greenplum安装包的时候,可能会发现不能下载,提示: 出现这个问题的原因有几个方面: 最常见的原因是注册账号是填写了虚假或者无意义的信息,譬如名字是 123,地址是 abc. Pivo ...

  8. SSM框架报错分析(一)——There is no getter for property named 'XXX' in 'class java.lang.String'

    一.发现问题 <select id="queryStudentByNum" resultType="student" parameterType=&quo ...

  9. Maven继承

    继承为了消除重复,可以把pom 中很多相同的配置提取出来:如:grouptId, version 等. 在使用的时候子工程直接继承父工程的依赖版本号,子工程中不再需要指定具体版本号,方便统一管控项目的 ...

  10. 【逆向工具】逆向工具101editor使用-游戏快速通关

    [渡者游戏简介] 船夫小江将运送客人的,羊.狐狸.草等物品过河,如果留下动物被其它种类吃掉任务就失败了.你需要帮助他做出正确的顺序选择.Ferryman是一款根据经典谜题改编的解谜游戏. 一.查看文件 ...