SpringBoot響應(yīng)處理之以Json數(shù)據(jù)返回的實(shí)現(xiàn)方法
一、示例代碼
訪問 localhost:8080/jsonTest —— 返回 json 格式的數(shù)據(jù)
@Controller
public class ResponseTestController {
@ResponseBody // 標(biāo)注 —— 自動返回json數(shù)據(jù)
@GetMapping("/jsonTest")
public Person testPerson(){
Person person = new Person();
person.setAge(20);
person.setUserName("Liuwanqing");
Pet pet = new Pet();
pet.setName("huahua");
pet.setAge("五個月");
person.setPet(pet);
return person;
}
}
二、返回值解析原理
SpringBoot 支持的返回值類型是由返回值解析器決定的,SpringBoot 返回值類型如下:
ModelAndView
Model
View
…
SpringBoot共含15種返回值解析器決定了其支持15種返回值:

三、源代碼分析(debug)
設(shè)置以下幾處斷點(diǎn):

step into —— 返回值處理器邏輯
this.returnValueHandlers.handleReturnValue(returnValue, this.getReturnValueType(returnValue), mavContainer, webRequest);
1. 先找返回值處理器
HandlerMethodReturnValueHandler handler = this.selectHandler(returnValue, returnType);
2. 執(zhí)行下列代碼,找到符合要求的返回值處理器
在眾多返回值處理器中找到符合要求的 —RequestResponseBodyMethodRrocessor
即 RequestResponseBodyMethodRrocessor 可處理標(biāo)注了 @ResponseBody 注解的返回值
private HandlerMethodReturnValueHandler selectHandler(@Nullable Object value, MethodParameter returnType) {
boolean isAsyncValue = this.isAsyncReturnValue(value, returnType);
Iterator var4 = this.returnValueHandlers.iterator();
HandlerMethodReturnValueHandler handler;
do {
do {
if (!var4.hasNext()) {
return null;
}
handler = (HandlerMethodReturnValueHandler)var4.next();
} while(isAsyncValue && !(handler instanceof AsyncHandlerMethodReturnValueHandler));
} while(!handler.supportsReturnType(returnType));
return handler;
}
3. 調(diào)用返回值處理器(interface HandlerMethodReturnValueHandler)處理:
判斷是否支持這種類型的返回值,支持調(diào)用返回值處理器,調(diào)用 handleReturnValue 進(jìn)行處理
public interface HandlerMethodReturnValueHandler {
boolean supportsReturnType(MethodParameter var1);
void handleReturnValue(@Nullable Object var1, MethodParameter var2, ModelAndViewContainer var3, NativeWebRequest var4) throws Exception;
}
4. RequestResponseBodyMethodRrocessor工作原理
關(guān)鍵代碼:
public void handleReturnValue(@Nullable Object returnValue, MethodParameter returnType, ModelAndViewContainer mavContainer, NativeWebRequest webRequest) throws IOException, HttpMediaTypeNotAcceptableException, HttpMessageNotWritableException {
mavContainer.setRequestHandled(true);
ServletServerHttpRequest inputMessage = this.createInputMessage(webRequest);
ServletServerHttpResponse outputMessage = this.createOutputMessage(webRequest);
this.writeWithMessageConverters(returnValue, returnType, inputMessage, outputMessage);
}
處理返回值的方法
使用消息轉(zhuǎn)換器(涉及內(nèi)容協(xié)商機(jī)制,請移步下文內(nèi)容協(xié)商機(jī)制)進(jìn)行寫出操作,消息轉(zhuǎn)換器工作過程如下:
- 瀏覽器告知服務(wù)器其處理能力
- 服務(wù)器根據(jù)其自身能力,判斷服務(wù)器能生產(chǎn)什么樣的內(nèi)容
- SpringMVC 遍歷所有容器的底層 HttpMessageConverter, 看誰能處理
四、內(nèi)容協(xié)商機(jī)制
1. 內(nèi)容協(xié)商
瀏覽器告知服務(wù)器其需要什么類型的服務(wù)類型
瀏覽器可接受的類型:

服務(wù)器得到瀏覽器的處理能力:

服務(wù)器將返回內(nèi)容轉(zhuǎn)為瀏覽器能處理的形式

2. 內(nèi)容協(xié)商原理重點(diǎn)源代碼
protected List<MediaType> getProducibleMediaTypes(HttpServletRequest request, Class<?> valueClass, @Nullable Type targetType) {
Set<MediaType> mediaTypes = (Set)request.getAttribute(HandlerMapping.PRODUCIBLE_MEDIA_TYPES_ATTRIBUTE);
if (!CollectionUtils.isEmpty(mediaTypes)) {
return new ArrayList(mediaTypes);
} else if (this.allSupportedMediaTypes.isEmpty()) {
return Collections.singletonList(MediaType.ALL);
} else {
List<MediaType> result = new ArrayList();
Iterator var6 = this.messageConverters.iterator();
while(true) {
while(var6.hasNext()) {
HttpMessageConverter<?> converter = (HttpMessageConverter)var6.next();
if (converter instanceof GenericHttpMessageConverter && targetType != null) {
if (((GenericHttpMessageConverter)converter).canWrite(targetType, valueClass, (MediaType)null)) {
result.addAll(converter.getSupportedMediaTypes());
}
} else if (converter.canWrite(valueClass, (MediaType)null)) {
result.addAll(converter.getSupportedMediaTypes());
}
}
return result;
}
}
}3. HttpMessageConverters消息轉(zhuǎn)換器原理
MessageConverter 規(guī)范

默認(rèn)的MessageConverter

最終 MappingJackson2HttpMessageConverter 把對象轉(zhuǎn)為JSON(利用底層的jackson的objectMapper轉(zhuǎn)換的)
總結(jié):
- 返回值處理器判斷是否支持這種類型的返回值( supportsReturnType)
- 返回值處理器調(diào)用handleReturnValue 進(jìn)行處理
- 以 @ResponseBody 注解為例,RequestResponseBodyMethodProcessor 可以處理返回值標(biāo)
- 最后,利用 MessageConverters 進(jìn)行處理 將數(shù)據(jù)寫為json
到此這篇關(guān)于SpringBoot響應(yīng)處理之以Json數(shù)據(jù)返回的實(shí)現(xiàn)方法的文章就介紹到這了,更多相關(guān)SpringBoot Json數(shù)據(jù)返回內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
spring?boot學(xué)習(xí)筆記之操作ActiveMQ指南
ActiveMQ是一種開源的基于JMS規(guī)范的一種消息中間件的實(shí)現(xiàn),ActiveMQ的設(shè)計目標(biāo)是提供標(biāo)準(zhǔn)的,面向消息的,能夠跨越多語言和多系統(tǒng)的應(yīng)用集成消息通信中間件,這篇文章主要給大家介紹了關(guān)于spring?boot學(xué)習(xí)筆記之操作ActiveMQ指南的相關(guān)資料,需要的朋友可以參考下2021-11-11
詳解java動態(tài)代理的2種實(shí)現(xiàn)方式
目前Java開發(fā)包中包含了對動態(tài)代理的支持,但是其實(shí)現(xiàn)只支持對接口的的實(shí)現(xiàn)。這篇文章主要介紹了詳解java動態(tài)代理的2種實(shí)現(xiàn)方式 ,有興趣的可以了解一下。2016-11-11
SpringBoot?SpringSecurity?詳細(xì)介紹(基于內(nèi)存的驗證)
這篇文章主要介紹了SpringBoot?SpringSecurity?介紹(基于內(nèi)存的驗證),本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-04-04
java使用JSCH實(shí)現(xiàn)SFTP文件管理
這篇文章主要為大家詳細(xì)介紹了java使用JSCH實(shí)現(xiàn)SFTP文件管理,實(shí)現(xiàn)上傳、下載等功能,具有一定的參考價值,感興趣的小伙伴們可以參考一下2019-08-08
java實(shí)現(xiàn)簡單汽車租賃系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)簡單汽車租賃系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-01-01

