SpringMVC restful 注解之@RequestBody進行json與object轉(zhuǎn)換
由于快過年的原因,項目組沒有太多任務,閑來無事研究了一下spring中restful調(diào)用。發(fā)現(xiàn)spring竟然已經(jīng)強大到如此境界,程序員已經(jīng)不需要在關心在寫接口的過程中數(shù)據(jù)的轉(zhuǎn)換以及調(diào)用,只需要專注業(yè)務。下面我總結(jié)一下步驟及其在研究過程的遇到的問題。
步驟:
1、git clone https://github.com/spring-guides/gs-rest-service.git 從spring官網(wǎng)上下載了源碼
2、進行maven編譯(gradle也行)
3、運行、訪問http://localhost:8080/greeting
4、運行結(jié)果能把對象轉(zhuǎn)換為json對象返回給頁面
這時我就在思考怎樣能讓請求的數(shù)據(jù)自動轉(zhuǎn)換為java對象呢,通過google,發(fā)現(xiàn)其實spring已經(jīng)提供了HttpMessageConverter轉(zhuǎn)換器,而且默認情況下是加載了 MappingJackson2HttpMessageConverter(json ~object轉(zhuǎn)換的類)。只需要配置@RequestBody Greeting gree 即可使用。
controller層代碼如下:
@RequestMapping(value = "/greeting", method = RequestMethod.POST,consumes = "application/json")
public @ResponseBody Greeting greeting(@RequestBody Greeting gree) {
System.out.println(gree.getContent());
return gree;
}
這時候我通過谷歌的插件(postman)進行調(diào)用,死活調(diào)用不成功!
分析問題及解決問題:
這時我感覺問題的原因可能出在如下幾個方面:
1、spring默認沒有加載MappingJackson2HttpMessageConverter(不知道具體加載方式)
2、MappingJackson2HttpMessageConverter加載后不能工作(不知道不工作原因)
其實最后面導致不工作的原因是太相信spring的源碼(對象沒有提供set方法導致),帶著這兩疑問在網(wǎng)上海量搜索者找不到對應結(jié)果。沒有辦法只能從根本上找到問題原因,看spring源代碼。
針對第一個問題:
第一步:手動重寫加載類型轉(zhuǎn)換器
@Configuration
@EnableWebMvc
public class WebConfiguration extends WebMvcConfigurerAdapter {
public void configureMessageConverters(List<HttpMessageConverter<?>> messageConverters) {
System.out.println("init convert is start !!!!!");
StringHttpMessageConverter stringConverter = new StringHttpMessageConverter();
stringConverter.setWriteAcceptCharset(false);
messageConverters.add(new MappingJackson2HttpMessageConverter());
System.out.println("init convert is stop !!!!!");
}
}
測試發(fā)現(xiàn)還是不能使用,這時就更不清楚原因了。只能看默認情況下spring是怎么加載類型轉(zhuǎn)換器的。結(jié)果發(fā)現(xiàn)在WebMvcConfigurationSupport中這個方法addDefaultHttpMessageConverters(HttpMessageConverter這個關鍵字反射搜索到使用地方通過判斷及其跟蹤找到的)中如下代碼:
@SuppressWarnings("deprecation")
protected final void addDefaultHttpMessageConverters(List<HttpMessageConverter<?>> messageConverters) {
StringHttpMessageConverter stringConverter = new StringHttpMessageConverter();
stringConverter.setWriteAcceptCharset(false);
messageConverters.add(new ByteArrayHttpMessageConverter());
messageConverters.add(stringConverter);
messageConverters.add(new ResourceHttpMessageConverter());
messageConverters.add(new SourceHttpMessageConverter<Source>());
messageConverters.add(new AllEncompassingFormHttpMessageConverter());
if (romePresent) {
messageConverters.add(new AtomFeedHttpMessageConverter());
messageConverters.add(new RssChannelHttpMessageConverter());
}
if (jaxb2Present) {
messageConverters.add(new Jaxb2RootElementHttpMessageConverter());
}
if (jackson2Present) {
messageConverters.add(new MappingJackson2HttpMessageConverter());
}
else if (jacksonPresent) {
messageConverters.add(new org.springframework.http.converter.json.MappingJacksonHttpMessageConverter());
}
}
已經(jīng)加載了相應的默認轉(zhuǎn)換器。斷點調(diào)試說明默認配置是沒有問題的。
只能說明是第二個問題導致的,但是不知道為什么導致這個問題(json數(shù)據(jù)問題,還是其他問題),在不知道問題的情況下,只能看request請求過來,轉(zhuǎn)換器是怎么工作的。因為本人對spring不是特別了解,所以不知其原理。在這種情況下還是只能根據(jù)(HttpMessageConverter)關鍵類找到相應使用地方。以經(jīng)驗進行判斷和調(diào)試。發(fā)現(xiàn)在AbstractMessageConverterMethodArgumentResolver中的readWithMessageConverters方法是request請求過來進行類型轉(zhuǎn)換的處理方法。
protected <T> Object readWithMessageConverters(HttpInputMessage inputMessage,
MethodParameter methodParam, Type targetType) throws IOException, HttpMediaTypeNotSupportedException {
MediaType contentType;
try {
contentType = inputMessage.getHeaders().getContentType();
}
catch (InvalidMediaTypeException ex) {
throw new HttpMediaTypeNotSupportedException(ex.getMessage());
}
if (contentType == null) {
contentType = MediaType.APPLICATION_OCTET_STREAM;
}
Class<?> contextClass = methodParam.getContainingClass();
Class<T> targetClass = (Class<T>) ResolvableType.forType(targetType,
ResolvableType.forMethodParameter(methodParam)).resolve();
for (HttpMessageConverter<?> converter : this.messageConverters) {
if (converter instanceof GenericHttpMessageConverter) {
GenericHttpMessageConverter<?> genericConverter = (GenericHttpMessageConverter<?>) converter;
if (genericConverter.canRead(targetType, contextClass, contentType)) {
if (logger.isDebugEnabled()) {
logger.debug("Reading [" + targetType + "] as \"" +
contentType + "\" using [" + converter + "]");
}
return genericConverter.read(targetType, contextClass, inputMessage);
}
}
if (targetClass != null) {
if (converter.canRead(targetClass, contentType)) {
if (logger.isDebugEnabled()) {
logger.debug("Reading [" + targetClass.getName() + "] as \"" +
contentType + "\" using [" + converter + "]");
}
return ((HttpMessageConverter<T>) converter).read(targetClass, inputMessage);
}
}
}
throw new HttpMediaTypeNotSupportedException(contentType, allSupportedMediaTypes);
}
這時候發(fā)現(xiàn)其實已經(jīng)根據(jù)HttpMessageConverter的canRead方法已經(jīng)找到了對應的類型消息轉(zhuǎn)換器MappingJackson2HttpMessageConverter,而且已經(jīng)開始進行轉(zhuǎn)換了,只是拋出了運行時異常。因為異常沒有在控制臺輸出。我通過斷點調(diào)試發(fā)現(xiàn)MappingJackson2HttpMessageConverter的readJavaType方法拋出運行時異常,通過源代碼發(fā)現(xiàn)底層是用的jackson的objectMapper進行操作的,代碼如下:
try {
return this.objectMapper.readValue(inputMessage.getBody(), javaType);
}
catch (IOException ex) {
throw new HttpMessageNotReadableException("Could not read JSON: " + ex.getMessage(), ex);
}
如是我就把代碼單獨拿出來在main方法里面運行,還是不行,這時我就好定位問題了。要不是類型錯誤,要不是輸入數(shù)據(jù)錯誤。仔細檢查發(fā)現(xiàn)json數(shù)據(jù)沒有問題,用jsonobject也能進行轉(zhuǎn)換。這時只能判斷是傳入的javaType有問題導致的。如是我打開發(fā)現(xiàn)對象(Greeting)沒有set方法,我想是不是因為此jakson沒法工作呢(原理不清楚)。如是乎我給此對象提供了set方法,再運行可以了。繞了一圈終于把問題解決了,但是通過這個問題讓我更加清楚了spring的restful的工作機制。
- 解讀@RequestBody的正確使用方法
- 快速解決SpringMVC @RequestBody 用map接收請求參數(shù)的問題
- 詳解SpringMVC @RequestBody接收Json對象字符串
- @ResponseBody 和 @RequestBody 注解的區(qū)別
- springmvc @RequestBody String類型參數(shù)的使用
- 關于Springboot | @RequestBody 接收到的參數(shù)對象屬性為空的問題
- Springboot攔截器如何獲取@RequestBody參數(shù)
- 親測SpringBoot參數(shù)傳遞及@RequestBody注解---踩過的坑及解決
- springMvc注解之@ResponseBody和@RequestBody詳解
- SpringBoot中@RequestBody的偽表單提交場景
相關文章
layui的select聯(lián)動實現(xiàn)代碼
今天小編就為大家分享一篇layui的select聯(lián)動實現(xiàn)代碼,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-09-09
Bootstrap CSS組件之下拉菜單(dropdown)
這篇文章主要為大家詳細介紹了Bootstrap CSS組件之下拉菜單(dropdown),具有一定的參考價值,感興趣的小伙伴們可以參考一下2016-12-12
移動端a標簽下載文件重命名(download)不生效解決辦法
在移動端使用a標簽下載文件時,文件重命名可能不生效,尤其是在APP內(nèi)嵌頁面中,這通常是因為跨域問題導致的,文中將解決辦法介紹的非常詳細,需要的朋友可以參考下2024-10-10
TypeScript中d.ts類型聲明文件的實現(xiàn)
.d.ts 文件是 TypeScript 的類型聲明文件,它們的主要作用是為 JavaScript 庫提供類型支持,本文主要介紹了TypeScript中d.ts類型聲明文件的實現(xiàn),感興趣的可以了解一下2023-10-10

