天天看點

Map<String,Object>接收參數,Long類型降級為Integer,報類型轉換異常

前言

今天看群裡小夥伴問了一個非常有意思的問題:

使用 Map<String,Object> 對象接收前端傳遞的參數,在後端取參時,因為接口文檔中明确該字段類型為 Long ,是以對接收的參數進行了強轉,即 (Long)參數 ,但是卻發生了類型轉換異常,報錯資訊如下:

class java.lang.Integer cannot be cast to class java.lang.Long 
(java.lang.Integer and java.lang.Long are in module java.base of loader 'bootstrap')
           

發現好幾個小夥伴也有疑惑,幹脆直接碼一篇文章解答一下,希望對有此疑惑的小夥伴有所幫助。

Long 類型降級

我們先通過測試資料、測試方法來還原一下問題。

測試json資料如下:

{
    "user_name": "niceyoo",
    "age": -24,
    "money": 2147483646
}
           

測試test方法如下:

@PostMapping("/test")
@ResponseBody
public void test(@RequestBody Map<String,Object> params) {
    String userName = (String) params.get("user_name");
    Integer age = (Integer) params.get("age");
    Long money = (Long) params.get("money");
    System.out.println(String.format("user_name=%s,age=%s,money=%s",userName,age,money));
}
           

如上傳遞了三個參數,即使用者名、年齡、金額這三個字段,如下是調用情況:

Map&lt;String,Object&gt;接收參數,Long類型降級為Integer,報類型轉換異常

通過截圖下方斷點參數可以看到接收的 age 、money 都是 Interger 類型,而代碼中 money 使用 Long 強轉的話會報 java.lang.Long cannot be cast to java.lang.Integer 異常,至此問題就還原出來了。

不懂就問:為什麼接收的 money 是 Integer 類型,強轉後會報錯?

首先我們先來看為什麼接收的 money 是 Integer 類型。

使用 Map<String,Object> 接收的 Long 數值如果處于 「 Integer.MIN_VALUE ~ Integer.MAX_VALUE 」 是會自動轉換成 Integer 的。

不光是接收,同樣直接使用 Map<String,Object> 存入資料符合這個範圍,仍然也會被認為存入的是 Integer 類型,我們可以把它看做一種潛在的優化,畢竟 Long 類型使用的位元組數要大于 Integer 。

  • Integer.MAX_VALUE,Integer 類型的最大值,
  • Integer.MIN_VALUE,Integer 類型的最小值,

這兩個值可以直接通過 sout 列印檢視:

  • System.out.println(Integer.MAX_VALUE): 2147483647
  • System.out.println(Integer.MIN_VALUE): -2147483648

我們來驗證一下,将 money 的值改成大于 Integer.MAX_VALUE:

Map&lt;String,Object&gt;接收參數,Long類型降級為Integer,報類型轉換異常

同樣的,如果傳遞的值小于 -2147483648,那麼同樣接收的類型為 Long,大家可以驗證一下。

為什麼強轉後會報錯?

強轉的一些條件:

  • 低階轉高階可以直接轉;
  • 高階轉低階需要強制轉換,否則報錯;
  • 強轉 null 值報錯;
  • 包裝類型資料不支援直接跨類型強轉;

其實報錯的原因就是這最後一條,包裝類型是不支援直接跨類型強轉的,比如,你可以使用 Integer 跟 int 的直接轉換,但是你不能将 Integer 直接強轉成 Long 類型,或者 Long 類型強轉 Integer ,這樣都是報錯的。

如果不能确定接收的對象是 Long 還是 Integer 怎麼辦?

既然知道不能強轉了,但是如果不知道接收的對象是什麼類型怎麼辦?

因為這種情況下,你不知道接收的對象到底是超過 Integer 這個範圍還是沒有超過。

第一種解決方法就是對接收的對象進行類型判斷。

主要就是使用關鍵字 — instanceof

Map&lt;String,Object&gt;接收參數,Long類型降級為Integer,報類型轉換異常

如圖所示,我們可以先用 Object 接收一下對象,然後對其通過 instanceof 關鍵字進行類型判斷,如果是 Integer 類型,則先 .toString(),然後再使用 Long.parseLong() 進行轉換,如果本身就是 Long,則直接進行強轉,同理,接收 Integer 類型也一樣,無非就是換用 Integer.parseInteger().

第二種就是直接使用實體接收,使用實體接收就不會存在此問題了。

Map&lt;String,Object&gt;接收參數,Long類型降級為Integer,報類型轉換異常
部落格園持續更新,歡迎關注,希望這篇文章對你有所幫助。

部落格園:https://www.cnblogs.com/niceyoo