一麻敌、現(xiàn)象
1.Accept值設(shè)置為application/json,或不設(shè)置值均可以返回正常JSON
{
"ResponseStatusObject": {
"Id": "1",
"LocalTime": "20191227141606",
"RequestURL": "/VIID/Subscribes/1",
"StatusCode": "7",
"StatusString": "JSON格式無(wú)效"
}
}
2.Accept設(shè)置為application/*+json后掂摔,返回的JSON中實(shí)體字段首字母變成小寫
{
"ResponseStatusObject": {
"id": "1",
"statusCode": "7",
"requestURL": "/VIID/Subscribes/1",
"statusString": "JSON格式無(wú)效",
"localTime": "20191227141437"
}
}
二术羔、原因
? 查看項(xiàng)目中是否配置了消息轉(zhuǎn)換器,比如配置了FastJsonHttpMessageConverter用來(lái)替換默認(rèn)的MappingJackson2HttpMessageConverter乙漓。
? 由于配置FastJsonHttpMessageConverter時(shí)只適配了application/json级历,所以Accept為/或application/json的可以通過(guò)此轉(zhuǎn)換器進(jìn)行轉(zhuǎn)換。如下:
// 添加支持的MediaTypes;不添加時(shí)默認(rèn)為*/*,也就是默認(rèn)支持全部
List<MediaType> fastMediaTypes = new ArrayList<>();
//這里只支持了application/json
fastMediaTypes.add(MediaType.APPLICATION_JSON_UTF8);
fastJsonHttpMessageConverter.setSupportedMediaTypes(fastMediaTypes);
? 而application/*+json由于沒(méi)有適配并且方法上加上了@ResponseBody注解叭披,所有還會(huì)走默認(rèn)的MappingJackson2HttpMessageConverter寥殖,所以會(huì)出現(xiàn)這種詭異的問(wèn)題。MappingJackson2HttpMessageConverter對(duì)pojo類型序列化的處理轉(zhuǎn)換邏輯如下:
Jackson2在初始化序列器時(shí)趋观,對(duì)pojo類型對(duì)象會(huì)收集其屬性信息扛禽,屬性包括成員變量及方法锋边,然后屬性名稱和處理過(guò)后的方法名稱做為key保存到一個(gè)LinkedHashMap中皱坛。收集過(guò)程中會(huì)調(diào)用com.fasterxml.jackson.databind.util.BeanUtil中的legacyManglePropertyName方法用來(lái)處理方法名稱,它會(huì)將get/set方法前綴豆巨,即get或set去掉剩辟,并將其后面的連續(xù)大寫字符轉(zhuǎn)換成小寫字符返回。例如: getNEWString會(huì)轉(zhuǎn)變成newstring返回。你的屬性名稱如果有這樣的"nSmallSellCount",lombok自動(dòng)生成的get方法就會(huì)是這樣的"getNSmallSellCount"贩猎,處理過(guò)后就是這樣的"nsmallSellCount",這與屬性nSmallSellCount并不沖突熊户,可以同時(shí)存在于HashMap中。收集完屬性信息后吭服,下面的步驟中會(huì)刪除掉非可見(jiàn)的屬性嚷堡,一般指的是私有成員變量,這時(shí)艇棕,名稱為"nSmallSellCount"的成員變量屬性會(huì)被刪除掉蝌戒,這樣的序列化結(jié)果是不會(huì)有問(wèn)題的,但沼琉,如果加了@JsonProperty注釋北苟,Jackson2會(huì)認(rèn)為這個(gè)屬性是可見(jiàn)的,不必會(huì)刪除打瘪,這時(shí)這兩個(gè)表示同一個(gè)值得屬性就會(huì)被一同序列化友鼻。
三、解決方案(兩種選擇其一即可)
? 1.在要轉(zhuǎn)換的POJO類中標(biāo)注@JsonProperty注解闺骚,指定想要轉(zhuǎn)換成的名字彩扔。
? 注意:此種方案有個(gè)問(wèn)題就是如果將注解標(biāo)注在字段上,那么轉(zhuǎn)成JSON后字段會(huì)出現(xiàn)重復(fù)問(wèn)題僻爽,具體原因可以參考上面的轉(zhuǎn)換邏輯借杰。將注解標(biāo)注在get\set方法上面即可。(如果使用了lombok进泼,就只能手動(dòng)編寫get\set方法了)
? 2.在配置的消息轉(zhuǎn)換器中增加對(duì)application/*+json的支持即可蔗衡。
MediaType mediaType = new MediaType("application", "viid+json", StandardCharsets.UTF_8);
fastMediaTypes.add(mediaType);