Spring MVC 解读——<mvc:annotation-driven/>
一、AnnotationDrivenBeanDefinitionParser
<mvc:annotation-driven/>标签,所以我们找到对应的实现类是org.springframework.web.servlet.config.AnnotationDrivenBeanDefinitionParser。
通过阅读类注释文档,我们发现这个类主要是用来向工厂中注册了
前两个是HandlerMapping接口的实现类,用来处理请求映射的。其中第一个是处理@RequestMapping注解的。第二个会将controller类的名字映射为请求url。
中间三个是用来处理请求的。具体点说就是确定调用哪个controller的哪个方法来处理当前请求。第一个处理@Controller注解的处理器,支持自定义方法参数和返回值(很酷)。第二个是处理继承HttpRequestHandler的处理器。第三个处理继承自Controller接口的处理器。
后面三个是用来处理异常的解析器。
HandlerMapping
Spring mvc 使用HandlerMapping来找到并保存url请求和处理函数间的mapping关系。
以DefaultAnnotationHandlerMapping为例来具体看HandlerMapping的作用
DefaultAnnotationHandlerMapping将扫描当前所有已经注册的spring beans中的@requestmapping标注以找出url 和 handler method处理函数的关系并予以关联。
Handleradapter
Spring MVC通过HandlerAdapter来实际调用处理函数。
以AnnotationMethodHandlerAdapter为例
DispatcherServlet中根据handlermapping找到对应的handler method后,首先检查当前工程中注册的所有可用的handlerAdapter,根据handlerAdapter中的supports方法找到可以使用的handlerAdapter。通过调用handlerAdapter中的handle方法来处理及准备handler method中的参数及annotation(这就是spring mvc如何将reqeust中的参数变成handle method中的输入参数的地方),最终调用实际的handle method。
public
BeanDefinition parse(Element element, ParserContext parserContext) {
Object source = parserContext.extractSource(element);
CompositeComponentDefinition compDefinition =
new
CompositeComponentDefinition(element.getTagName(), source);
parserContext.pushContainingComponent(compDefinition);
RuntimeBeanReference contentNegotiationManager = getContentNegotiationManager(element, source, parserContext);
//第一个在这 RequestMappingHandlerMapping
RootBeanDefinition handlerMappingDef =
new
RootBeanDefinition(RequestMappingHandlerMapping.
class
);
handlerMappingDef.setSource(source);
handlerMappingDef.setRole(BeanDefinition.ROLE_INFRASTRUCTURE);
handlerMappingDef.getPropertyValues().add(
"order"
,
0
);
handlerMappingDef.getPropertyValues().add(
"removeSemicolonContent"
,
false
);
handlerMappingDef.getPropertyValues().add(
"contentNegotiationManager"
, contentNegotiationManager);
String methodMappingName = parserContext.getReaderContext().registerWithGeneratedName(handlerMappingDef);
//第二个在这 RequestMappingHandlerAdapter
RootBeanDefinition handlerAdapterDef =
new
RootBeanDefinition(RequestMappingHandlerAdapter.
class
);
handlerAdapterDef.setSource(source);
handlerAdapterDef.setRole(BeanDefinition.ROLE_INFRASTRUCTURE);
handlerAdapterDef.getPropertyValues().add(
"contentNegotiationManager"
, contentNegotiationManager);
handlerAdapterDef.getPropertyValues().add(
"webBindingInitializer"
, bindingDef);
handlerAdapterDef.getPropertyValues().add(
"messageConverters"
, messageConverters);
if
(element.hasAttribute(
"ignoreDefaultModelOnRedirect"
)) {
Boolean ignoreDefaultModel = Boolean.valueOf(element.getAttribute(
"ignoreDefaultModelOnRedirect"
));
handlerAdapterDef.getPropertyValues().add(
"ignoreDefaultModelOnRedirect"
, ignoreDefaultModel);
}
if
(argumentResolvers !=
null
) {
handlerAdapterDef.getPropertyValues().add(
"customArgumentResolvers"
, argumentResolvers);
}
if
(returnValueHandlers !=
null
) {
handlerAdapterDef.getPropertyValues().add(
"customReturnValueHandlers"
, returnValueHandlers);
}
if
(asyncTimeout !=
null
) {
handlerAdapterDef.getPropertyValues().add(
"asyncRequestTimeout"
, asyncTimeout);
}
if
(asyncExecutor !=
null
) {
handlerAdapterDef.getPropertyValues().add(
"taskExecutor"
, asyncExecutor);
}
handlerAdapterDef.getPropertyValues().add(
"callableInterceptors"
, callableInterceptors);
handlerAdapterDef.getPropertyValues().add(
"deferredResultInterceptors"
, deferredResultInterceptors);
String handlerAdapterName = parserContext.getReaderContext().registerWithGeneratedName(handlerAdapterDef);
//异常处理解析器
RootBeanDefinition exceptionHandlerExceptionResolver =
new
RootBeanDefinition(ExceptionHandlerExceptionResolver.
class
);
exceptionHandlerExceptionResolver.setSource(source);
exceptionHandlerExceptionResolver.setRole(BeanDefinition.ROLE_INFRASTRUCTURE);
exceptionHandlerExceptionResolver.getPropertyValues().add(
"contentNegotiationManager"
, contentNegotiationManager);
exceptionHandlerExceptionResolver.getPropertyValues().add(
"messageConverters"
, messageConverters);
exceptionHandlerExceptionResolver.getPropertyValues().add(
"order"
,
0
);
String methodExceptionResolverName =
parserContext.getReaderContext().registerWithGeneratedName(exceptionHandlerExceptionResolver);
//异常处理解析器
RootBeanDefinition responseStatusExceptionResolver =
new
RootBeanDefinition(ResponseStatusExceptionResolver.
class
);
responseStatusExceptionResolver.setSource(source);
responseStatusExceptionResolver.setRole(BeanDefinition.ROLE_INFRASTRUCTURE);
responseStatusExceptionResolver.getPropertyValues().add(
"order"
,
1
);
String responseStatusExceptionResolverName =
parserContext.getReaderContext().registerWithGeneratedName(responseStatusExceptionResolver);
//异常处理解析器
RootBeanDefinition defaultExceptionResolver =
new
RootBeanDefinition(DefaultHandlerExceptionResolver.
class
);
defaultExceptionResolver.setSource(source);
defaultExceptionResolver.setRole(BeanDefinition.ROLE_INFRASTRUCTURE);
defaultExceptionResolver.getPropertyValues().add(
"order"
,
2
);
String defaultExceptionResolverName =
parserContext.getReaderContext().registerWithGeneratedName(defaultExceptionResolver);
parserContext.registerComponent(
new
BeanComponentDefinition(handlerMappingDef, methodMappingName));
parserContext.registerComponent(
new
BeanComponentDefinition(handlerAdapterDef, handlerAdapterName));
parserContext.registerComponent(
new
BeanComponentDefinition(exceptionHandlerExceptionResolver, methodExceptionResolverName));
parserContext.registerComponent(
new
BeanComponentDefinition(responseStatusExceptionResolver, responseStatusExceptionResolverName));
parserContext.registerComponent(
new
BeanComponentDefinition(defaultExceptionResolver, defaultExceptionResolverName));
parserContext.registerComponent(
new
BeanComponentDefinition(mappedCsInterceptorDef, mappedInterceptorName));
//这里注册了BeanNameUrlHandlerMapping,SimpleControllerHandlerAdapter等
// Ensure BeanNameUrlHandlerMapping (SPR-8289) and default HandlerAdapters are not "turned off"
MvcNamespaceUtils.registerDefaultComponents(parserContext, source);
parserContext.popAndRegisterContainingComponent();
return
null
;
}
//在这啊。
public
static
void
registerDefaultComponents(ParserContext parserContext, Object source) {
registerBeanNameUrlHandlerMapping(parserContext, source);
registerHttpRequestHandlerAdapter(parserContext, source);
registerSimpleControllerHandlerAdapter(parserContext, source);
三、总结
我们知道了它们自动为我们注册了这么多的Bean,那这些Bean是做什么的呢?
我们主要说明里面的两个,RequestMappingHandlerMapping和RequestMappingHandlerAdapter。
第一个是HandlerMapping的实现类,它会处理@RequestMapping 注解,并将其注册到请求映射表中。(下片文章我们会详细介绍的)
第二个是HandlerAdapter的实现类,它是处理请求的适配器,说白了,就是确定调用哪个类的哪个方法,并且构造方法参数,返回值。(后面文章也会陆续详细介绍的)
那么它跟<context:component-scan/>有什么区别呢?其实想上篇文章中介绍的,<context:component-scan/>标签是告诉Spring 来扫描指定包下的类,并注册被@Component,@Controller,@Service,@Repository等注解标记的组件。
而<mvc:annotation-scan/>是告知Spring,我们启用注解驱动。然后Spring会自动为我们注册上面说到的几个Bean到工厂中,来处理我们的请求。
<context:annotation-config/>是对包进行扫描,实现注释驱动Bean定义,同时将bean自动注入容器中使用。即解决了@Controller标识的类的bean的注入和使用。
Spring MVC分发请求到控制器所必须的DefaultAnnotationHandlerMapping和AnnotationMethodHandlerAdapter实例