返回值解决
上一节讲到了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 讲完,开瓶茅台庆贺一下~~