日本综合一区二区|亚洲中文天堂综合|日韩欧美自拍一区|男女精品天堂一区|欧美自拍第6页亚洲成人精品一区|亚洲黄色天堂一区二区成人|超碰91偷拍第一页|日韩av夜夜嗨中文字幕|久久蜜综合视频官网|精美人妻一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時(shí)間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營(yíng)銷解決方案
為什么阿里巴巴Java開發(fā)手冊(cè)中強(qiáng)制要求超大整數(shù)禁止使用Long類型返回?

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

創(chuàng)新互聯(lián)建站是一家專注于成都做網(wǎng)站、成都網(wǎng)站建設(shè)、成都外貿(mào)網(wǎng)站建設(shè)和多線BGP機(jī)房的網(wǎng)絡(luò)公司,有著豐富的建站經(jīng)驗(yàn)和案例。

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

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

錯(cuò)誤演示

創(chuàng)建一個(gè) Spring Boot 項(xiàng)目,然后在新建一個(gè)接口,可以返回 DbScript 對(duì)象,其中 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.     @RequestMapping("/info")
  6.     public DbScript getDbScript() {
  7.         DbScript dbScript = new DbScript();
  8.         // 賦予一個(gè)大整數(shù) long 型腳本 id
  9.         long id = IdWorker.getId();
  10.         dbScript.setId(id);
  11.         logger.info("id:{}", id);
  12.         return dbScript;
  13.     }
  14. }

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

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

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

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

解決方法

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

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

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

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

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

第一種方法

第一種方法比較簡(jiǎn)單,直接將 Long id; 改為 String id;,這種只適用于這個(gè)對(duì)象只在這個(gè)方法中使用了,比較局限。

第二種方法

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

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

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

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

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

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

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

第三種方法

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

總結(jié)

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


新聞標(biāo)題:為什么阿里巴巴Java開發(fā)手冊(cè)中強(qiáng)制要求超大整數(shù)禁止使用Long類型返回?
文章來源:http://www.dlmjj.cn/article/dpisecj.html