av激情亚洲男人的天堂国语,日韩欧美精品一中文字幕,无码av一区二区三区无码,国产又色又爽又刺激的a片,国产又色又爽又刺激的a片

誰動(dòng)了我的代碼——Long精度丟失

一個(gè)詭異的現(xiàn)象

在進(jìn)行數(shù)據(jù)結(jié)構(gòu)設(shè)計(jì)時(shí),我們通常需要考慮到相關(guān)業(yè)務(wù)的數(shù)據(jù)量等因素。比如非核心業(yè)務(wù)但數(shù)據(jù)量大并且頻繁寫入的表的主鍵,我們可能會(huì)考慮設(shè)計(jì)為Long類型。剛開始,數(shù)據(jù)量小,可能并不會(huì)發(fā)現(xiàn)什么問題。但是當(dāng)數(shù)據(jù)量大了,或者Id采用雪花算法生成,這個(gè)時(shí)候詭異的事情便會(huì)發(fā)生。

后端數(shù)據(jù)正常返回,postman調(diào)試看數(shù)據(jù)也正常。但是當(dāng)前端用后端返回的這個(gè)id查詢相應(yīng)的數(shù)據(jù)時(shí),便會(huì)發(fā)生詭異的NotFoundException,或者查詢的出來的數(shù)據(jù)和原先的數(shù)據(jù)不一致。

所以,誰偷偷動(dòng)了我的代碼?

JavaScript的數(shù)值精度

如果只從后端分析問題,或者只從前端分析問題,那永遠(yuǎn)也找不到答案。

在 JavaScript 中,數(shù)值類型默認(rèn)會(huì)被轉(zhuǎn)換為雙精度浮點(diǎn)數(shù),而雙精度浮點(diǎn)數(shù)的精度有限,只能精確表示 2 的 53 次方以內(nèi)(即 Number.MAX_SAFE_INTEGER,約為 9 x 10^15)的整數(shù)。對(duì)于超過該范圍的長整數(shù),JavaScript 會(huì)發(fā)生精度丟失,導(dǎo)致值變得不準(zhǔn)確。

例如一個(gè)雪花算法生成的ID 1734042308679487490,前端獲取到的值卻變成了1734042308679487500

知道了問題的原因,問題就容易解決了——將Long類型作為String類型返回給前端即可。

一個(gè)簡(jiǎn)單的解決辦法

(1) Spring Boot 中提供了 @JsonFormat 注解,可以對(duì)實(shí)體類中的屬性進(jìn)行序列化和反序列化格式化。對(duì)于 Long 類型的屬性,可以設(shè)置其格式為字符串類型,并在前端進(jìn)行相應(yīng)的處理,以保持其精度不丟失。如:

public class Order {
  @JsonFormat(shape = JsonFormat.Shape.STRING)
  private Long id;
  ...
}

前端獲取到的是string類型的數(shù)據(jù),自然也不會(huì)有精度丟失的問題了。

(2) SpringBoot也支持在通過配置文件在項(xiàng)目級(jí)別,將數(shù)值類型的數(shù)據(jù)轉(zhuǎn)成字符串返回給前端,通過在 application.properties 文件中添加配置即可:

# 默認(rèn)為false
spring.jackson.seralization.WRITE_NUMBER_AS_STRINGS=true

(3) 如果不想使用 @JsonFormat 注解或者項(xiàng)目不是基于SpringBoot框架構(gòu)建的,同樣的思路,直接將Long類型轉(zhuǎn)換成String返回給前端即可。

總結(jié)

在 JavaScript 中數(shù)值類型最大精度大約為9*10^15,即超過16位的數(shù)值一定會(huì)存在精度丟失問題。因此,后端返回Long類型的數(shù)值時(shí),需要轉(zhuǎn)換成String給到前端。


分享標(biāo)題:誰動(dòng)了我的代碼——Long精度丟失
文章鏈接:http://uogjgqi.cn/article/dpcgocs.html
掃二維碼與項(xiàng)目經(jīng)理溝通

我們?cè)谖⑿派?4小時(shí)期待你的聲音

解答本文疑問/技術(shù)咨詢/運(yùn)營咨詢/技術(shù)建議/互聯(lián)網(wǎng)交流