DispatcherServlet.init执行调用链
- init()
- initServletBean()
- initWebApplicationContext()
- createWebApplicationContext(rootContext)
- createWebApplicationContext(rootContext)
- createWebApplicationContext((ApplicationContext) parent)
- configureAndRefreshWebApplicationContext(wac)
- wac.refresh() ;//其中wac=XmlWebApplicationContext
- configureAndRefreshWebApplicationContext(wac)
- createWebApplicationContext((ApplicationContext) parent)
- createWebApplicationContext(rootContext)
- createWebApplicationContext(rootContext)
- initWebApplicationContext()
- initServletBean()
- 可以看到DispatcherServlet 在执行初始化逻辑时,会调用容器启动加载bean逻辑。
XmlWebApplicationContext继承关系图
image.png- 对Spring 容器启动过程有一定了解的同学一定知道,在refresh()方法执行过程中,首先会去解析指定的配置文件,完成bean定义的加载。
- 本文将重点放在解析mvc标签。
一份配置demo
dispatcher-servlet.xml.png-
自定义元素节点spring-mvc的解析逻辑,容器上下文首先会获取对应节点元素指定的NamespaceHandler实现类(MvcNamespaceHandler),反射加载类对象,并执行init逻辑,自定义元素的解析交给init逻辑中初始化进去的解析器处理
DefaultNamespaceHandlerResolver-resolve.png
MvcNamespaceHandler.init()执行逻辑
namespaceHandler-init.png- 从初始化的解析器列表来看,对照配置demo,重点关注annotation-driven元素匹配的解析器AnnotationDrivenBeanDefinitionParser。
AnnotationDrivenBeanDefinitionParser.parse(参数)执行逻辑
- 注册RequestMappingHandlerMapping的bean定义到DefaultListableBeanFactory对象中;
- RequestMappingHandlerMapping处理请求映射的,处理@RequestMapping跟请求地址之间的关系。
- 注册RequestMappingHandlerAdapter的bean定义到DefaultListableBeanFactory对象中。
- RequestMappingHandlerAdapter是请求处理的适配器,也就是请求之后处理具体逻辑的执行,关系到哪个类的哪个方法以及转换器等工作。
RequestMappingHandlerMapping类继承关系
RequestMappingHandlerMapping.png- 由RequestMappingHandlerMapping的类继承图可以看到,当RequestMappingHandlerMapping被懒加载时,会执行afterPropertiesSet()方法,具体重写逻辑在AbstractHandlerMethodMapping的方法initHandlerMethods内部。
AbstractHandlerMethodMapping.initHandlerMethods执行逻辑
AbstractHandlerMethodMapping.initHandlerMethods.png- 获取Spring 初始化的所以bean;
- 循环遍历beanNames,根据bean的class文件判断该bean是否包含@Controller或者@RequestMapping注解;
如果包含注解@Controller、@RequestMapping,则执行detectHandlerMethods(beanName)逻辑; - handlerMethodsInitialized(getHandlerMethods());默认无实现逻辑。
AbstractHandlerMethodMapping.detectHandlerMethods的执行逻辑
AbstractHandlerMethodMapping.detectHandlerMethods.png- 根据请求参数传过来的beanName,获取对应bean的Class对象。
- 循环遍历出bean的所有方法,并调用T result = metadataLookup.inspect(specificMethod)方法创建RequestMappingInfo对象,将方法名称和RequestMappingInfo对象已键值对的形式保存在局部变量methods中;
- 最后通过beanName、方法、@RequestMapping注解合并后对应的RequestMappingInfo对象,完成HandlerMethod的注册。
RequestMappingHandlerMapping.getMappingForMethod执行逻辑
RequestMappingHandlerMapping.getMappingForMethod.png- 尝试创建参数传递过来带有@RequestMapping注解的方法对应的RequestMappingInfo对象;
- 如果RequestMappingInfo对象不为null,尝试创建带有@RequestMapping注解的Controller类对应的RequestMappingInfo对象;
- 如果类对应的RequestMappingInfo对象不为null,则合并类对应RequestMappingInfo对象和方法对应的RequestMappingInfo对象。
AbstractHandlerMethodMapping.registerHandlerMethod的执行逻辑
- 调用属性mappingRegistry(内部类)的register(T mapping, Object handler, Method method)
MappingRegistry.register的执行逻辑
mappingRegistry.register.png- 根据beanName和方法创建HandlerMethod对象;
- 将RequestMappingInfo对象和HandlerMethod对象以键值对的形式放入属性mappingLookup内;
- 将@RequestMapping注解合并的url和RequestMappingInfo对象以键值对的形式放入属性urlLookup内;
- 至此两个比较重要的属性就已经构造完成。代码中其他部分就略过。
到这总结一下
- 注册RequestMappingHandlerMapping的懒加载对象实例化后,调用了afterPropertiesSet()方法,最终在afterPropertiesSet()方法执行完毕后将带有@RequestMapping注解的java 类的相关信息设置到实例化之后的RequestMappingHandlerMapping对象中。也就是说相关请求对应的Controller类和方法的匹配条件在bean被加载完成后就已准备好。所以结合DispatcherServlet处理请求的执行逻辑(doDispatch的执行流程)。
RequestMappingHandlerAdapter类继承关系
RequestMappingHandlerAdapter.png- 由RequestMappingHandlerAdapter的类继承图可以看到,当RequestMappingHandlerAdapter被懒加载时,会执行afterPropertiesSet()方法。
RequestMappingHandlerAdapter.afterPropertiesSet执行逻辑
RequestMappingHandlerAdapter.afterPropertiesSet.png- 初始化请求参数解析器argumentResolvers,初始化的值使用框架默认。
- 初始化返回结果处理器returnValueHandlers,初始化的值使用框架默认。
网友评论