返回值處理
上一節(jié)講到了DispatcherServlet 根據(jù)訪問(wèn)路徑找到對(duì)應(yīng)的類方法讳窟,在調(diào)用了類方法后根时,需要做返回值的處理仰迁,比如返回json等。打開(kāi)源碼耐薯,開(kāi)干~
時(shí)光逆流~回到 doDispatch( )方法舔清,找到這行碼:
// Actually invoke the handler.
mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
別慫,點(diǎn)進(jìn)去曲初,一直找到 RequestMappingHandlerAdapter#handleInternal()方法体谒,又看到一行核心代碼:
// No synchronization on session demanded at all...
mav = invokeHandlerMethod(request, response, handlerMethod);
還沒(méi)到,再點(diǎn)進(jìn)去臼婆,又找到一行核心代碼:
invocableMethod.invokeAndHandle(webRequest, mavContainer);
打開(kāi)該方法后抒痒,上馬:
public void invokeAndHandle(ServletWebRequest webRequest, ModelAndViewContainer mavContainer,Object... providedArgs) throws Exception {
// 對(duì)請(qǐng)求進(jìn)行處理,解析入?yún)⒛慷В祷爻鰠? Object returnValue = invokeForRequest(webRequest, mavContainer, providedArgs);
……
try {
// 對(duì)返回值進(jìn)行處理
this.returnValueHandlers.handleReturnValue(
returnValue, getReturnValueType(returnValue), mavContainer, webRequest);
}
……
}
點(diǎn)擊this.returnValueHandlers.handleReturnValue()進(jìn)去:
@Override
public void handleReturnValue(Object returnValue, MethodParameter,returnType, ModelAndViewContainer mavContainer, NativeWebRequest webRequest){
// 查找返回值的處理器
HandlerMethodReturnValueHandler handler = selectHandler(returnValue, returnType);
……
// 處理了返回值
handler.handleReturnValue(returnValue, returnType, mavContainer,
webRequest);
}
看一眼 selectHandler()的私人照:
可以看到评汰,這里是通過(guò)遍歷所有的handler,找到與returnType匹配的handler痢虹,我們看一下 handler.supportsReturnType(returnType)是怎么處理的被去,比如遍歷到 RequestResponseBodyMethodProcessor,打開(kāi)它的supportsReturnType()方法:
@Override
public boolean supportsReturnType(MethodParameter returnType) {
return (AnnotatedElementUtils.hasAnnotation(returnType.getContainingClass(),
ResponseBody.class) || returnType.hasMethodAnnotation(ResponseBody.class));
}
這里也是很簡(jiǎn)單奖唯,不就是判斷有沒(méi)有 @ResponseBody這個(gè)注解咩惨缆,有的話就返回這個(gè)處理器。
回到 handleReturnValue()方法丰捷,看那一行代碼:
handler.handleReturnValue(returnValue, returnType, mavContainer, webRequest);
進(jìn)入坯墨,上馬
@Override
public void handleReturnValue(@Nullable Object returnValue, MethodParameter returnType, ModelAndViewContainer mavContainer, NativeWebRequest webRequest)
throws IOException, HttpMediaTypeNotAcceptableException, HttpMessageNotWritableException {
mavContainer.setRequestHandled(true);
// 構(gòu)建輸入輸出流
ServletServerHttpRequest inputMessage = createInputMessage(webRequest);
ServletServerHttpResponse outputMessage = createOutputMessage(webRequest);
// 轉(zhuǎn)換信息并寫出
writeWithMessageConverters(returnValue, returnType, inputMessage, outputMessage);
}
點(diǎn)擊 writeWithMessageConverters()進(jìn)入,代碼很多病往,上有用的:
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);
……
// 返回值轉(zhuǎn)換
if (genericConverter != null) {
genericConverter.write(body, targetType, selectedMediaType,
outputMessage);
}
}
}
最后找到的是 AbstractJackson2HttpMessageConverter#writeInternal()捣染,該方法內(nèi)的:
objectWriter.writeValue(generator, value);
這行代碼,會(huì)調(diào)用jackson工具把輸出值進(jìn)行json序列化了停巷。
再到后面是model和view的處理耍攘,都什么年代了榕栏,大部分項(xiàng)目都是前后端分離了,這個(gè)就跳過(guò)了蕾各。
springmvc 講完扒磁,開(kāi)瓶茅臺(tái)慶祝一下~~