當JSON解析遇上了泛型,該如何處理泛型擦除問題
JSON是一種輕量級的數據交換格式,簡潔和清晰的層次結構使得 JSON 成為理想的數據交換語言,常被用于實際項目中。Java生態圈中有很多處理JSON格式化的類庫,如json-lib框架、Jackson、Google的Gson、阿里的FastJson等,通過這些類庫可以使我們更加簡便地處理JSON。下面本文主要通過一個實際案例討論分析在解析JSON字符串的過程中遇到泛型該如何處理。
- 案例描述
- 問題引出
- 處理方案
- 原理分析
一、案例描述
首先介紹三個類,為了便于理解將代碼省去非關鍵部分。
- PrepCommonResp 是一個通用響應類,為了支持多種返回類型,設置了其中一個屬性result為泛型。
- public class PrepCommonResp<T> {
- private T result; //獲取調用返回值
- private String code = "000000"; //獲取錯誤碼
- private String msg = "Success";
- //這里省略getter setter等
- }
- public class LeaderboardResp implements Serializable {
- List<LeaderboardPojo> leaderboardList;
- //這里省略getter setter等
- }
- public class LeaderboardPojo {
- //基本屬性
- }
需求:將一個JSON字符串的String類型數據解析到指定了泛型的通用響應類實例PrepCommonResp
二、問題引出
一開始編寫時,采用了Jackson的ObjectMapper類readValue方法來進行解析,具體代碼如下。
- ObjectMapper mapper = new ObjectMapper();
- PrepCommonResp<LeaderboardResp> resp = mapper.readValue(result,PrepCommonResp.class);
我們期待的結果是JSON字符串被正確解析,并且result屬性的值是以指定泛型LeaderboardResp的格式存儲。但是通過斷點之后我們發現result屬性的值是以LinkedHashMap的形式存儲,與期望結果不符。
三、解決方案
在分析原因之前,先來說說解決方式。解決方式有多種,我們通過了嘗試對比了各種方案之后選擇了其中一種較為簡便的就是采用Gson的fromJson方法來解決,具體代碼如下:
- //嘗試使用Gson
- Gson gson = new Gson();
- PrepCommonResp<LeaderboardResp> resp = gson.fromJson(result, new TypeToken<PrepCommonResp<LeaderboardResp>>(){}.getType());
- List<LeaderboardPojo> list = resp.getResult().getLeaderboardList();
運行結果如下,result的值以我們期望的LeaderboardResp形式存儲
四、原理分析
1. 先理解泛型與編譯器虛擬機的關系以及什么是擦除?
解析: Java語言的泛型基本上是在編譯器中實現的。由編譯器執行類型檢測和推斷后生成普通的非泛型的字節,虛擬機是完全無感知泛型存在的,這種實現技術稱為擦除。編譯器使用泛型類型信息保證類型安全,然后在生成字節碼之前將其清除。
2. 為什么使用Jackson的readValue方法解析JSON字符串后result的屬性值會是LinkedHashMap而不是我們指定的LeaderboardResp呢?
解析:泛型只在編譯期間起到檢測作用,當編譯器將泛型類編譯完成之后,泛型類的類型參數都被全部擦除。接下來在運行期間虛擬機并不知道泛型的存在,當對JSON字符串進行解析時由于泛型被擦除了導致虛擬機并不知道要將其解析成哪種類型,所以就解析為了默認的LinkedHashMap類型,導致出現了上面的場景。
3. 那Gson是如何解決泛型擦除這種情況呢?
我們來看看這一句核心代碼
- PrepCommonResp<LeaderboardResp> resp = gson.fromJson(result, new TypeToken<PrepCommonResp<LeaderboardResp>>(){}.getType());
解析:Gson的做法非常巧妙,如上面的代碼所示,將需要獲取類型的泛型類作為TypeToken的泛型參數構造一個匿名的子類,然后通過getType()方法就可以獲取到我們想要的泛型類的泛型參數類型??梢岳斫鉃槭菍⒎盒皖愋痛嫫饋?,解決了泛型擦除的問題。
五、最后
感謝您的閱讀,如果喜歡本文歡迎關注和轉發,本頭條號將持續分享IT技術知識。對于文章內容有其他想法或意見建議等,歡迎提出共同討論共同進步。如果您對于此場景有更好的解決方案也歡迎提出討論。