前言
今天看群里小伙伴問了一個非常有意思的問題:
使用 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().
第二種就是直接使用實體接收,使用實體接收就不會存在此問題了。
博客園持續更新,歡迎關注,希望這篇文章對你有所幫助。