Map接收參數,Long類型降級爲Integer,報類型轉換異常

轉自:https://www.cnblogs.com/niceyoo/p/14008466.html

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

使用 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));
}

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

通過截圖下方斷點參數可以看到接收的 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:

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

爲什麼強轉後會報錯?

強轉的一些條件:

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

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

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

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

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

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

主要就是使用關鍵字 — instanceof

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

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

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章