返回值解决

上一节讲到了DispatcherServlet 依据拜访门路找到对应的类办法,在调用了类办法后,须要做返回值的解决,比方返回json等。关上源码,开干~

时光顺流~回到 doDispatch( )办法,找到这行码:

// Actually invoke the handler.mv = ha.handle(processedRequest, response, mappedHandler.getHandler());

别怂,点进去,始终找到 RequestMappingHandlerAdapter#handleInternal()办法,又看到一行外围代码:

// No synchronization on session demanded at all...mav = invokeHandlerMethod(request, response, handlerMethod);

还没到,再点进去,又找到一行外围代码:

invocableMethod.invokeAndHandle(webRequest, mavContainer);

关上该办法后,下马:

public void invokeAndHandle(ServletWebRequest webRequest, ModelAndViewContainer mavContainer,Object... providedArgs) throws Exception {    // 对申请进行解决,解析入参,返回出参    Object returnValue = invokeForRequest(webRequest, mavContainer, providedArgs);    ……    try {       // 对返回值进行解决        this.returnValueHandlers.handleReturnValue(        returnValue, getReturnValueType(returnValue), mavContainer, webRequest);    }    ……}

点击this.returnValueHandlers.handleReturnValue()进去:

@Overridepublic void handleReturnValue(Object returnValue, MethodParameter,returnType, ModelAndViewContainer mavContainer, NativeWebRequest webRequest){    // 查找返回值的处理器    HandlerMethodReturnValueHandler handler = selectHandler(returnValue, returnType);    ……   // 解决了返回值   handler.handleReturnValue(returnValue, returnType, mavContainer,    webRequest);}

看一眼 selectHandler()的私人照:

能够看到,这里是通过遍历所有的handler,找到与returnType匹配的handler,咱们看一下 handler.supportsReturnType(returnType)是怎么解决的,比方遍历到 RequestResponseBodyMethodProcessor,关上它的supportsReturnType()办法:

@Overridepublic boolean supportsReturnType(MethodParameter returnType) {    return (AnnotatedElementUtils.hasAnnotation(returnType.getContainingClass(),       ResponseBody.class) || returnType.hasMethodAnnotation(ResponseBody.class));}

这里也是很简略,不就是判断有没有 @ResponseBody这个注解咩,有的话就返回这个处理器。

回到 handleReturnValue()办法,看那一行代码:

handler.handleReturnValue(returnValue, returnType, mavContainer, webRequest);

进入,下马

@Overridepublic void handleReturnValue(@Nullable Object returnValue, MethodParameter returnType,    ModelAndViewContainer mavContainer, NativeWebRequest webRequest)throws IOException, HttpMediaTypeNotAcceptableException, HttpMessageNotWritableException {    mavContainer.setRequestHandled(true);    // 构建输入输出流    ServletServerHttpRequest inputMessage = createInputMessage(webRequest);    ServletServerHttpResponse outputMessage = createOutputMessage(webRequest);    // 转换信息并写出    writeWithMessageConverters(returnValue, returnType, inputMessage, outputMessage);    }

点击 writeWithMessageConverters()进入,代码很多,上有用的:

for (HttpMessageConverter<?> converter : this.messageConverters) {    GenericHttpMessageConverter genericConverter = (converter instanceof GenericHttpMessageConverter ?(GenericHttpMessageConverter<?>) converter : null);    // canWrite()判断获取 genericConverter    if (genericConverter != null ?        ((GenericHttpMessageConverter) converter).canWrite(targetType, valueType, selectedMediaType) :converter.canWrite(valueType, electedMediaType))         {        body = getAdvice().beforeBodyWrite(body, returnType, selectedMediaType,            (Class<? extends HttpMessageConverter<?>>) converter.getClass(),            inputMessage, outputMessage);        ……        // 返回值转换        if (genericConverter != null) {            genericConverter.write(body, targetType, selectedMediaType,                                    outputMessage);        }    }}

最初找到的是 AbstractJackson2HttpMessageConverter#writeInternal(),该办法内的:

objectWriter.writeValue(generator, value);

这行代码,会调用jackson工具把输入值进行json序列化了。

再到前面是model和view的解决,都什么年代了,大部分我的项目都是前后端拆散了,这个就跳过了。

springmvc 讲完,开瓶茅台庆贺一下~~