一、工作机制
在容器初始化时会建立所有url和controller的对应关系,保存到Map<url,controller>中。tomcat启动时会通知spring初始化容器(加载bean的定义信息和初始化所有单例bean),然后springmvc会遍历容器中的bean,获取每一个controller中的所有方法访问的url,然后将url和controller保存到一个Map中。
当请求到来是,根据request的URL快速定位到controller,因为最终处理request的是controller中的方法,Map中只保留了url和controller中的对应关系,这一步工作的原理就是拼接controller的URL(controller上@RequestMapping的值)和方法的url(method上@RequestMapping的值),与request的url进行匹配,找到匹配的那个方法;
确定处理请求的method后,接下来的任务就是参数绑定,把request中参数绑定到方法的形式参数上,这一步是整个请求处理过程中最复杂的一个步骤(后面会详细讲)。springmvc提供了两种request参数与方法形参的绑定方法:
① 通过注解进行绑定,@RequestParam
② 通过参数名称进行绑定。
使用注解进行绑定,我们只要在方法参数前面声明@RequestParam("xxx"),就可以将request中参数xxx的值绑定到方法的该参数上。使用参数名称进行绑定的前提是必须要获取方法中参数的名称,Java反射只提供了获取方法的参数的类型,并没有提供获取参数名称的方法。springmvc解决这个问题的方法是用asm框架读取字节码文件,来获取方法的参数名称.asm框架是一个字节码操作框架。
最后,解析方法和参数上的注解,最后反射调用方法获取ModelAndView结果视图。
二、总结,主要三个步骤
第一步、建立Map<urls,controller>的关系
我们首先看第一个步骤,也就是建立Map<url,controller>关系的部分。第一部分的入口类为ApplicationObjectSupport的setApplicationContext方法,setApplicationContext方法中核心部分就是初始化容器initApplicationContext(context),子类AbstractDetectingUrlHandlerMapping实现了该方法。
第二步、根据访问url找到对应controller中处理请求的方法.
第二个步骤是由请求触发的,所以入口为DispatcherServlet.DispatcherServlet的核心方法为doService(),doService()中的核心逻辑由doDispatch()实现。
第三步、反射调用处理请求的方法,返回结果视图
根据url确定controller中处理请求的方法,然后通过反射获取该方法上的注解和参数,解析方法和参数上的注解,最后反射调用方法获取ModelAndView结果视图。因为上面采用注解url形式说明的,所以我们这里继续以注解处理器适配器来说明,调用的就是AnnotationMethodHandlerAdapter的handle()。handle()中的核心逻辑由invokeHandlerMethod(request, response, handler)实现。