Jackson序列化丟失泛型的解決
Jackson序列化丟失泛型
經(jīng)過(guò)
項(xiàng)目中遇到一個(gè)奇怪的bug,即一個(gè)Map<Integer,List<Integer>>的泛型map,向map中g(shù)et一個(gè)存在的key,事實(shí)上卻返回null。
經(jīng)過(guò)排查,發(fā)現(xiàn)是該map被Jackson序列化后,key的類型從Integer變成了String類型。再經(jīng)過(guò)反序列化,即使已經(jīng)聲明key泛型的Integer,反序列化后內(nèi)存數(shù)據(jù)中的key為String并不是Integer類型且并未拋出異常。
復(fù)現(xiàn)
1、聲明一個(gè)key泛型為Integer的map
Map<Integer, List<Integer>> map = new HashMap<>(); map.put(1, Arrays.asList(1,2,3)); map.put(1001,Arrays.asList(4,5,6)); map.put(50001,Arrays.asList(7,8,9));
2、申明Jackson序列化工具
ObjectMapper om = new ObjectMapper(); om.setVisibility(JsonMethod.ALL, JsonAutoDetect.Visibility.ANY); om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
3、序列化
String json = om.writeValueAsString(map); System.out.println(json);
4、序列化輸出
["java.util.HashMap",{"1":["java.util.ArrayList",[1,2,3]],"50001":["java.util.ArrayList",[7,8,9]],"1001":["java.util.ArrayList",[4,5,6]]}]
5、反序列化
Map<Integer,List<Integer>> map2 = om.readValue(json, Map.class); System.out.println(map2);
6、反序列化輸出
{1=[1, 2, 3], 50001=[7, 8, 9], 1001=[4, 5, 6]}
分析
由步驟4見(jiàn)得Map<Integer,List<Integer>>序列化后,key的Integer泛型已經(jīng)丟失,類型由Integer變?yōu)榱薙tring。
且步驟6反序列化后,盡管map的key申明為Integer類型,但是Jackson反序列化后,依然將key反序列化為String類型,且未拋出任何異常。此時(shí)通過(guò)Integer的key獲取map對(duì)應(yīng)的值永遠(yuǎn)返回null。
解決
對(duì)于可以指定返回類型的反序列化,可以通過(guò)Jackson的API指定反序列化對(duì)象的泛型。
Map<Integer, List<Integer>> map3 = om.readValue(json, new TypeReference<Map<Integer, List<Integer>>>(){}); System.out.println(map3);
對(duì)于通用型序列化反序列化的場(chǎng)景,例如RedisTemplate的序列化反序列化工具,無(wú)法指定特定的反序列化對(duì)象泛型,可以考慮使用其他序列化工具替代Jackson例如Fastjson。
序列化后反序列化丟失幾大問(wèn)題總結(jié)
序列化 (Serialization)將對(duì)象的狀態(tài)信息轉(zhuǎn)換為可以存儲(chǔ)或傳輸?shù)男问降倪^(guò)程。在序列化期間,對(duì)象將其當(dāng)前狀態(tài)寫入到臨時(shí)或持久性存儲(chǔ)區(qū)。
以后,可以通過(guò)從存儲(chǔ)區(qū)中讀取或反序列化對(duì)象的狀態(tài),重新創(chuàng)建該對(duì)象。
反序列化失敗原因:(目前只遇到過(guò)兩種)
沒(méi)有添加 serialVersionUID 可能會(huì)導(dǎo)致反序列化失敗
生成默認(rèn)的serialVersionUID --> [Add default serial version ID]
例如:
private static final long serialVersionUID = 1L;
生成串行serialVersionUID --> [Add generated serial version ID]
例如:
private static final long serialVersionUID =-5666638870709238304L;
注解生成serialVersionUID --> [Add @SuppressWarnings serial to serialVersionUID]
例如:
@SuppressWarnings("person") public class Person implements Serializable {}
繼承了一個(gè)已經(jīng)實(shí)現(xiàn)序列化接口的父類
并且與父類有重復(fù)的屬性,在反序列化的時(shí)候就會(huì)導(dǎo)致重復(fù)的屬性數(shù)據(jù)丟失
然后還有第三種就是使用Spring框架的情況下
如果布爾類型的對(duì)象屬性名以is開(kāi)頭,在序列化的時(shí)候會(huì)導(dǎo)致該屬性值丟失
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
java去除空格、標(biāo)點(diǎn)符號(hào)的方法實(shí)例
這篇文章主要給大家介紹了關(guān)于java去除空格、標(biāo)點(diǎn)符號(hào)的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09深入詳解Java中synchronized鎖升級(jí)的套路
synchronized鎖是啥?鎖其實(shí)就是一個(gè)對(duì)象,隨便哪一個(gè)都可以,Java中所有的對(duì)象都是鎖,換句話說(shuō),Java中所有對(duì)象都可以成為鎖。本文我們主要來(lái)聊聊synchronized鎖升級(jí)的套路,感興趣的可以收藏一下2023-04-04十分鐘速懂java知識(shí)點(diǎn) System類
這篇文章主要介紹了java知識(shí)點(diǎn)System類,根據(jù)一次面試總結(jié)的,可以十分鐘速懂System類,感興趣的小伙伴們可以參考一下2015-12-12spring boot-2.1.16整合swagger-2.9.2 含yml配置文件的代碼詳解
這篇文章主要介紹了spring boot-2.1.16整合swagger-2.9.2 含yml配置文件,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-08-08淺析RxJava處理復(fù)雜表單驗(yàn)證問(wèn)題的方法
這篇文章主要介紹了RxJava處理復(fù)雜表單驗(yàn)證問(wèn)題的相關(guān)資料,非常不錯(cuò)具有參考借鑒價(jià)值,需要的朋友可以參考下2016-06-06springboot如何連接兩個(gè)數(shù)據(jù)庫(kù)(多個(gè))
這篇文章主要介紹了springboot如何連接兩個(gè)數(shù)據(jù)庫(kù)(多個(gè)),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-01-01idea進(jìn)程結(jié)束但是項(xiàng)目頁(yè)面正常運(yùn)行怎么辦
這篇文章主要介紹了idea進(jìn)程結(jié)束但是項(xiàng)目頁(yè)面正常運(yùn)行怎么辦,很多朋友遇到這樣的情況不知道該如何解決了,下面小編給大家?guī)?lái)了idea進(jìn)程結(jié)束但是項(xiàng)目頁(yè)面正常運(yùn)行的解決方法,需要的朋友可以參考下2023-03-03Java后端對(duì)接微信支付(小程序、APP、PC端掃碼)包含查單退款
微信支付我們主要聚焦于這三種支付方式,其中JSPAI與APP主要與uniapp開(kāi)發(fā)微信小程序與APP對(duì)接,本文主要介紹了Java后端對(duì)接微信支付(小程序、APP、PC端掃碼)包含查單退款,具有一定的參考價(jià)值,感興趣的可以了解一下2021-12-12解決使用@RequestParam注解和泛型遇到的問(wèn)題
這篇文章主要介紹了解決使用@RequestParam注解和泛型遇到的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10