源码-spring接收参数原理

一、DispatchServlet简介

servlet依托于容器,处理请求及相应的运行在支持java语言服务器的小程序。目前最常见的就是用来扩展Java Web服务器功能。

二、接收参数过程



HandlerMethodReturnValueHandler:响应返回值的处理

HandlerMethodArgumentResolver:对应请求方法参数的处理

public interface HandlerMethodArgumentResolver {

##该解析器是否支持parameter参数的解析

boolean supportsParameter(MethodParameter var1);

//将方法参数解析为参数值并返回

    Object resolveArgument(MethodParameter var1, ModelAndViewContainer var2, NativeWebRequest var3, WebDataBinderFactory var4)throws Exception;

}

HandlerMethodArgumentResolver的初始化:bean生命周期afterPropertiesSet方法初始化参数解析器:

public void afterPropertiesSet() {

// Do this first, it may add ResponseBody advice beans

  initControllerAdviceCache();

//初始化SpringMVC默认的方法参数解析器,并添加至argumentResolvers

  if (this.argumentResolvers ==null) {

List resolvers = getDefaultArgumentResolvers();

      this.argumentResolvers =new HandlerMethodArgumentResolverComposite().addResolvers(resolvers);

  }

if (this.initBinderArgumentResolvers ==null) {

//初始化SpringMVC默认的初始化绑定器参数解析器

List resolvers = getDefaultInitBinderArgumentResolvers();

      this.initBinderArgumentResolvers =new HandlerMethodArgumentResolverComposite().addResolvers(resolvers);

  }

if (this.returnValueHandlers ==null) {

List handlers = getDefaultReturnValueHandlers();

      this.returnValueHandlers =new HandlerMethodReturnValueHandlerComposite().addHandlers(handlers);

  }

}

进入getDefalutArgumentResolvers方法,代码如下:

private List getDefaultArgumentResolvers() {

List resolvers =new ArrayList();

  // Annotation-based argument resolution基于注解

//一般用于带有@RequestParam注解的简单参数绑定,简单参数比如byte、int、long、double、String以及对应的包装类型

  resolvers.add(new RequestParamMethodArgumentResolver(getBeanFactory(), false));

//用于处理带有@RequestParam注解,且参数类型为Map的解析绑定

  resolvers.add(new RequestParamMapMethodArgumentResolver());

//一般用于处理带有@PathVariable注解的默认参数绑定,后续还有一些参数解析器,可以通过以下解析器的supportsParameter(MethodParameter parameter)方法查看

  resolvers.add(new PathVariableMethodArgumentResolver());

  resolvers.add(new PathVariableMapMethodArgumentResolver());

  resolvers.add(new MatrixVariableMethodArgumentResolver());

  resolvers.add(new MatrixVariableMapMethodArgumentResolver());

  resolvers.add(new ServletModelAttributeMethodProcessor(false));

//带有@RequestBody注解的

   resolvers.add(new RequestResponseBodyMethodProcessor(getMessageConverters(), this.requestResponseBodyAdvice));

  resolvers.add(new RequestPartMethodArgumentResolver(getMessageConverters(), this.requestResponseBodyAdvice));

  resolvers.add(new RequestHeaderMethodArgumentResolver(getBeanFactory()));

  resolvers.add(new RequestHeaderMapMethodArgumentResolver());

  resolvers.add(new ServletCookieValueMethodArgumentResolver(getBeanFactory()));

  resolvers.add(new ExpressionValueMethodArgumentResolver(getBeanFactory()));

  // Type-based argument resolution  基于类型

  resolvers.add(new ServletRequestMethodArgumentResolver());

  resolvers.add(new ServletResponseMethodArgumentResolver());

  resolvers.add(new HttpEntityMethodProcessor(getMessageConverters(), this.requestResponseBodyAdvice));

  resolvers.add(new RedirectAttributesMethodArgumentResolver());

  resolvers.add(new ModelMethodProcessor());

  resolvers.add(new MapMethodProcessor());

  resolvers.add(new ErrorsMethodArgumentResolver());

  resolvers.add(new SessionStatusMethodArgumentResolver());

  resolvers.add(new UriComponentsBuilderMethodArgumentResolver());

  // Custom arguments

  if (getCustomArgumentResolvers() !=null) {

resolvers.addAll(getCustomArgumentResolvers());

  }

// Catch-all

  resolvers.add(new RequestParamMethodArgumentResolver(getBeanFactory(), true));

  resolvers.add(new ServletModelAttributeMethodProcessor(true));

  return resolvers;

}

HandlerMethodArgumentResolverComposite:实际的参数解析类

//上面存放的默认的HandlerMethodArgumentResolver参数解析器

private final List<HandlerMethodArgumentResolver> argumentResolvers =new LinkedList();

//存放已经解析过的参数,已经对应的HandlerMethodArgumentResolver解析器

private final Map<MethodParameter, HandlerMethodArgumentResolver> argumentResolverCache =new ConcurrentHashMap(256);

ServletInvocableHandlerMethod类:

public final Object invokeForRequest(NativeWebRequest request,

                                        ModelAndViewContainer mavContainer,

                                        Object... providedArgs) throws Exception {

        //从request中解析出HandlerMethod方法所需要的参数,并返回Object[]        Object[] args = getMethodArgumentValues(request, mavContainer, providedArgs);

        if (logger.isTraceEnabled()) {

            StringBuilder builder = new StringBuilder("Invoking [");

            builder.append(this.getMethod().getName()).append("] method with arguments ");

            builder.append(Arrays.asList(args));

            logger.trace(builder.toString());

        }

        //通过反射执行HandleMethod中的method,方法参数为args。并返回方法执行的返回值        Object returnValue = invoke(args);

        if (logger.isTraceEnabled()) {

            logger.trace("Method [" + this.getMethod().getName() + "] returned [" + returnValue + "]");

        }

        return returnValue;

    }

直接进入getMethodArgumentValues方法看看其过程,代码如下:

/**

* Get the method argument values for the current request.

*/

private Object[]getMethodArgumentValues(NativeWebRequest request, ModelAndViewContainer mavContainer,

      Object... providedArgs)throws Exception {

//获取方法参数数组

MethodParameter[] parameters = getMethodParameters();

//创建一个参数数组,保存从request解析出的方法参数

  Object[] args =new Object[parameters.length];

  for (int i =0; i < parameters.length; i++) {

MethodParameter parameter = parameters[i];

      parameter.initParameterNameDiscovery(this.parameterNameDiscoverer);

      GenericTypeResolver.resolveParameterType(parameter, getBean().getClass());

      args[i] = resolveProvidedArgument(parameter, providedArgs);

      if (args[i] !=null) {

continue;

      }

//判断之前RequestMappingHandlerAdapter初始化的那24个HandlerMethodArgumentResolver(参数解析器),是否存在支持该参数解析的解析器

if (this.argumentResolvers.supportsParameter(parameter)) {

try {

args[i] =this.argumentResolvers.resolveArgument(

parameter, mavContainer, request, this.dataBinderFactory);

continue;

        }

catch (Exception ex) {

if (logger.isDebugEnabled()) {

logger.debug(getArgumentResolutionErrorMessage("Error resolving argument", i), ex);

            }

throw ex;

        }

}

if (args[i] ==null) {

String msg = getArgumentResolutionErrorMessage("No suitable resolver for argument", i);

        throw new IllegalStateException(msg);

      }

}

return args;

}

进入HandlerMethodArgumentResolverComposite的resolveArgument方法:

public ObjectresolveArgument(MethodParameter parameter, ModelAndViewContainer mavContainer,

      NativeWebRequest webRequest, WebDataBinderFactory binderFactory)throws Exception {

HandlerMethodArgumentResolver resolver = getArgumentResolver(parameter);

  if (resolver ==null) {

throw new IllegalArgumentException("Unknown parameter type [" + parameter.getParameterType().getName() +"]");

  }

return resolver.resolveArgument(parameter, mavContainer, webRequest, binderFactory);

}

然后进入不同的HandlerMethodArgumentResolver的resolverArgument方法

三、不同类型参数代码debug

四、自定义参数解析器

1、自定义参数注解

@Target(ElementType.PARAMETER)

@Retention(RetentionPolicy.RUNTIME)

public @interface ArgumentResolverAnnotation {

Stringvalue()default "";

}

2、自定义参数解析器实现HandlerMethodArgumentResolver接口

public class MyArgumentResolverimplements HandlerMethodArgumentResolver {

/**

* 解析器是否支持当前参数

    * @param parameter

    * @return

    */

    @Override

    public boolean supportsParameter(MethodParameter parameter) {

return parameter.hasParameterAnnotation(ArgumentResolverAnnotation.class);

    }

/**

* 解析参数

    * @param parameter

    * @param mavContainer

    * @param webRequest

    * @param binderFactory

    * @return

    * @throws Exception

*/

    @Override

    public ObjectresolveArgument(MethodParameter parameter, ModelAndViewContainer mavContainer, NativeWebRequest webRequest, WebDataBinderFactory binderFactory)throws Exception {

//new参数对象

        Object obj = BeanUtils.instantiate(parameter.getParameterType());

        BeanWrapper wrapper = PropertyAccessorFactory.forBeanPropertyAccess(obj);

        Iterator paramNames = webRequest.getParameterNames();

        while (paramNames.hasNext()) {

String paramName = paramNames.next();

            Object o = webRequest.getParameter(paramName);

            try {

wrapper.setPropertyValue(paramName, o);

            }catch (BeansException e) {

}

}

return obj;

    }

}

3、注册到Spring参数解析器中

<mvc:annotation-driven>


    <mvc:argument-resolvers>

        <bean class="com.chen.example.argumentResolver.MyArgumentResolver"/>

    </mvc:argument-resolvers>

</mvc:annotation-driven>

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,406评论 6 503
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,732评论 3 393
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,711评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,380评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,432评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,301评论 1 301
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,145评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,008评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,443评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,649评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,795评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,501评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,119评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,731评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,865评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,899评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,724评论 2 354

推荐阅读更多精彩内容