自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

為什么阿里巴巴Java開發(fā)手冊中強(qiáng)制要求超大整數(shù)禁止使用Long類型返回?

開發(fā) 后端
創(chuàng)建一個 Spring Boot 項目,然后在新建一個接口,可以返回 DbScript 對象,其中 id 是由 mybatis-plus 的 IdWorker.getId(基于 Snowflake 算法)生成的 19 位 long 類型的數(shù)值。

[[341915]]

本文轉(zhuǎn)載自微信公眾號「武培軒」,作者武培軒 。轉(zhuǎn)載本文請聯(lián)系武培軒公眾號。

 在閱讀《阿里巴巴Java開發(fā)手冊》時,發(fā)現(xiàn)有一條關(guān)于前后端超大整數(shù)返回的規(guī)約,具體內(nèi)容如下:

這個問題在之前和前端聯(lián)調(diào)的時候發(fā)生過,發(fā)現(xiàn)根據(jù)腳本 id 去審批的時候,狀態(tài)沒有變化,后來和前端溝通后,才知道這是 JavaScript 的一個坑,下面來復(fù)現(xiàn)下這個錯誤:

錯誤演示

創(chuàng)建一個 Spring Boot 項目,然后在新建一個接口,可以返回 DbScript 對象,其中 id 是由 mybatis-plus 的 IdWorker.getId(基于 Snowflake 算法)生成的 19 位 long 類型的數(shù)值。

  1. @RestController 
  2. @RequestMapping("/dbScrip"
  3. public class DbScriptController { 
  4.     Logger logger = LoggerFactory.getLogger(DbScriptController.class); 
  5.  
  6.     @RequestMapping("/info"
  7.     public DbScript getDbScript() { 
  8.         DbScript dbScript = new DbScript(); 
  9.         // 賦予一個大整數(shù) long 型腳本 id 
  10.         long id = IdWorker.getId(); 
  11.         dbScript.setId(id); 
  12.         logger.info("id:{}", id); 
  13.         return dbScript; 
  14.     } 

接著啟動服務(wù),在瀏覽器上訪問該接口,結(jié)果如下所示:

通過日志可以看到后端傳給前端的 id 為 1304270071757017088,但是前端拿到的卻為 1304270071757017000,其中發(fā)生了精度損失。

為什么會發(fā)生這樣的情況呢?

通過開發(fā)手冊,我們可以知道如果返回的數(shù)值超過 2 的 53 次方,就會轉(zhuǎn)換成 JS 的 Number,此時有些數(shù)值就有可能發(fā)生精度損失。

解決方法

那如果遇到了這種情況,該如何解決呢?

不要慌,可以采取以下幾種方法:

如果這個對象只在這個方法中用到了,可以將該屬性直接從 Long 類型改為 String 類型。

如果這個對象在很多地方都用到了,可以在序列化的時候,將 Long 類型轉(zhuǎn)換成 String 類型。

還可以添加一個新的 String 類型的屬性,專門用來在前后端傳輸這種大整數(shù)。

第一種方法

第一種方法比較簡單,直接將 Long id; 改為 String id;,這種只適用于這個對象只在這個方法中使用了,比較局限。

第二種方法

第二種方法可以在屬性上增加注解,如果使用的Jackson,可以添加 @JsonFormat(shape = JsonFormat.Shape.STRING) 或者 @JsonSerialize(using = ToStringSerializer.class) 注解。

如果這種需要修改的情況比較多,那么逐個添加還是有點(diǎn)費(fèi)事,那么還有什么好辦法嗎?

如果使用的是Jackson,它有個配置參數(shù) WRITE_NUMBERS_AS_STRINGS,可以強(qiáng)制將所有數(shù)字全部轉(zhuǎn)成字符串輸出,使用方法很簡單,只需要配置參數(shù)即可:spring.jackson.generator.write_numbers_as_strings=true,這種方式的優(yōu)點(diǎn)是使用方便,不需要調(diào)整代碼;缺點(diǎn)是顆粒度太大,所有的數(shù)字都被轉(zhuǎn)成字符串輸出了,包括按照 timestamp 格式輸出的時間也是如此。

那么還有什么方法能夠只對 Long 類型進(jìn)行處理轉(zhuǎn)換成 String 類型呢?

Jackson 提供了這種支持,可以對 ObjectMapper 進(jìn)行定制,具體代碼如下所示:

  1. public class JacksonConfiguration { 
  2.  
  3.     @Bean 
  4.     public Jackson2ObjectMapperBuilderCustomizer jackson2ObjectMapperBuilderCustomizer() { 
  5.         return jacksonObjectMapperBuilder -> jacksonObjectMapperBuilder 
  6.                 .serializerByType(Long.class, ToStringSerializer.instance) 
  7.                 .serializerByType(Long.TYPE, ToStringSerializer.instance); 
  8.     } 

通過定義 Jackson2ObjectMapperBuilderCustomizer,對 Jackson2ObjectMapperBuilder 對象進(jìn)行定制,對 Long 型數(shù)據(jù)進(jìn)行了定制,使用ToStringSerializer來進(jìn)行序列化。

第三種方法

第三種方法就需要多一個屬性,比如使用String dbScripId,用來代替之前的 id。

總結(jié)

本文針對《阿里巴巴Java開發(fā)手冊》中的對于需要使用超大整數(shù)的場景,服務(wù)端一律使用 String 字符串類型返回,禁止使用Long 類型出發(fā),提出了幾種解決方法,大家可以根據(jù)自己的需求去選擇方法,有其他解決方法的也歡迎留言討論。

 

責(zé)任編輯:武曉燕 來源: 武培軒
相關(guān)推薦

2021-10-11 09:32:40

包裝類型屬性

2025-04-17 08:47:23

2019-06-26 07:54:53

ArrayListsubList源碼

2022-03-14 09:41:10

POJO類型系統(tǒng)

2023-04-03 07:03:51

阿里巴巴List元素

2020-09-08 16:25:18

Apache BeancopyJava

2020-07-30 12:16:33

阿里巴巴Apache對象

2022-09-05 10:06:21

MySQL外循環(huán)內(nèi)循環(huán)

2017-05-02 21:14:20

阿里巴巴Java開發(fā)

2020-09-22 11:40:53

BigDecimalequalsJava

2019-09-02 15:20:28

Java開發(fā)繼承

2019-09-04 11:02:54

繼承層次組合

2021-10-20 14:53:31

Foreach強(qiáng)制阿里巴巴

2018-10-16 15:34:17

阿里巴巴Apache Flin大數(shù)據(jù)

2021-09-17 06:28:20

JOIN阿里Java

2021-08-04 17:20:30

阿里巴巴AsyncJava

2022-08-30 16:38:30

阿里巴巴JavaLog4j

2013-08-22 09:26:38

去IOE王堅

2018-12-29 15:41:41

阿里巴巴程序員serialVersi

2016-09-21 20:28:55

阿里巴巴IOE
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號