共计 3160 个字符,预计需要花费 8 分钟才能阅读完成。
返回值解决
上一节讲到了 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() 进去:
@Override
public 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() 办法:
@Override
public boolean supportsReturnType(MethodParameter returnType) {return (AnnotatedElementUtils.hasAnnotation(returnType.getContainingClass(),
ResponseBody.class) || returnType.hasMethodAnnotation(ResponseBody.class));
}
这里也是很简略,不就是判断有没有 @ResponseBody 这个注解咩,有的话就返回这个处理器。
回到 handleReturnValue() 办法,看那一行代码:
handler.handleReturnValue(returnValue, returnType, mavContainer, webRequest);
进入,下马
@Override
public 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 讲完,开瓶茅台庆贺一下~~