首先,我们先回忆下日常开发中的骚操作,比如请求一个url /product?id=xxx&name=yyy, 再controol中的对应映射方法,只要填一个id的参数,就能得到xxx的值,又比如我想得到request的值,直接填个request,它就给我了,仿佛springmvc要啥就有啥,非常的神奇,下面就看下源码是如何实现的。
在知道springmvc外围方法运行原理(即service方法及调用链三层以内...)的基础上,一个请求是由handlerMapping找到对应的handler及intercepts交给despatcherServlet去叫handlerAdpter执行,handler的执行的过程是在handlerAdpter中。这个参数解析的过程我可以明确的判断是在,handlerAdpter去执行具体的handler之间发生的
那么是怎么实现的呢?假如要我做一个这个功能我觉得很简单,在从handlerMapping中找到的handler,它里面肯定封装了对应处理器类的Class,方法的method,那么我只要拿到method根据它的paramter的类型,去找我目前能不能提供这个参数。找到了把请求param的String类型进行转换。赋值给方法的参数即可。但是springmvc中可以获得那么多类型,而且每种类型的解析方式都是不同的(比如request是直接提供的,String name是从request的参数中取的,Model好像是new出来的 等等),假如能够做好这个功能,大量的逻辑判断写到一起,扩展性也极差。
下面我们来看下springmvc是怎么做的。
一、首先既然是handlerAdpter干活的,我们就来看下一般我们用的RequestMappingHandlerAdapter的类长什么样
public class RequestMappingHandlerAdapter extends AbstractHandlerMethodAdapter
implements BeanFactoryAware, InitializingBean { //实现了initializingBean,初始化会调afterPropertiesSet
private List<HandlerMethodArgumentResolver> customArgumentResolvers; //看名字叫自定义参数解析
private HandlerMethodArgumentResolverComposite argumentResolvers; //看名字叫参数解析的组合
private HandlerMethodArgumentResolverComposite initBinderArgumentResolvers; //initBinder的参数解析(暂不分析全局的影响)
private List<HandlerMethodReturnValueHandler> customReturnValueHandlers; //看名字叫自定返回值的处理
private HandlerMethodReturnValueHandlerComposite returnValueHandlers; //返回值处理的组合
private List<ModelAndViewResolver> modelAndViewResolvers;
1.1、从上图我们知道此adater它首先可以拿到BeanFactory,然后会调用一个初始化方法,它的成员变量包括一些参数解析和返回值解析的数组,以及一个叫做xx组合的。下面就看下初始化的方法(构造函数主要构建了messageConverters(@XXBody注解用的),本次不研究这个)。
public void afterPropertiesSet() {
if (this.argumentResolvers == null) {
List<HandlerMethodArgumentResolver> resolvers = getDefaultArgumentResolvers();
this.argumentResolvers = new HandlerMethodArgumentResolverComposite().addResolvers(resolvers);
}
if (this.initBinderArgumentResolvers == null) {
List<HandlerMethodArgumentResolver> resolvers = getDefaultInitBinderArgumentResolvers();
this.initBinderArgumentResolvers = new HandlerMethodArgumentResolverComposite().addResolvers(resolvers);
}
if (this.returnValueHandle