中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

SpringBoot雪花算法主鍵ID傳到前端后精度丟失問題如何解決

發布時間:2022-08-09 17:21:43 來源:億速云 閱讀:257 作者:iii 欄目:開發技術

這篇文章主要介紹“SpringBoot雪花算法主鍵ID傳到前端后精度丟失問題如何解決”,在日常操作中,相信很多人在SpringBoot雪花算法主鍵ID傳到前端后精度丟失問題如何解決問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”SpringBoot雪花算法主鍵ID傳到前端后精度丟失問題如何解決”的疑惑有所幫助!接下來,請跟著小編一起來學習吧!

問題描述

Java后端Long類型的范圍

  • -2^63~2^63,即:-9223372036854775808~9223372036854775807,它是19位的。

  • 這個數字可以通過方法獲得:Long.MAX_VALUE、Long_MIN_VALUE。

前端JS的數字類型的范圍

  • -2^53~2^53,即:-9007199254740991~9007199254740991,它是16位的。

  • 這個數字可以通過方法獲得:Number.MAX_SAFE_INTEGER、Number.MIN_SAFE_INTEGER。

結論

可見,Java后端的Long寬度大于前端的。雪花算法一般會生成18位或者19位寬度的數字,那么這時就會出問題。

項目場景

1.表結構

主鍵類型是BIGINT,存儲雪花算法生成的ID。

CREATE TABLE `user` (
  `id` BIGINT(32) NOT NULL COMMENT '用戶id',
	...
  PRIMARY KEY (`id`) USING BTREE
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='用戶表';

2.Entity

用Long 類型對應數據庫ID的BIGINT類型。

這里使用 MybatisPlus 的雪花算法自動生成19位長度的純數字作為主鍵ID。(當然也可以手動用雪花算法生成ID)

import lombok.Data;
 
@Data
public class User {
    @TableId(type = IdType.ASSIGN_ID)
    private Long id;
	
    //其他成員
}

3.響應給前端

以JSON數據響應給前端正常

{
  "id": 1352166380631257089,
   ...
}

問題描述

實例

Controller

package com.knife.controller;
 
import com.knife.entity.UserVO;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
 
@RestController
@RequestMapping("user")
public class UserController {
 
    @GetMapping("find")
    public UserVO find(Long id) {
        UserVO userVO = new UserVO();
        userVO.setId(id);
        userVO.setUsername("Tony");
 
        return userVO;
    }
}

Entity

package com.knife.entity;
 
import lombok.Data;
 
@Data
public class UserVO {
    private Long id;
 
    private String username;
}

測試

訪問:http://localhost:8080/user/find?id=1352213368413982722

結果

SpringBoot雪花算法主鍵ID傳到前端后精度丟失問題如何解決

問題復現

從上邊可以看到,并沒有問題。

為什么沒有出問題?

前端傳入后端:SpingMVC會自動將String類型的ID轉為Long類型,不會出問題后端響應給前端:是JSON格式,與JS沒有關系,不會出問題

什么時候會出問題?

前端接收到JSON之后,將其序列化為JS對象,然后進行其他操作。在JSON轉JS對象時就會出問題,如下:

SpringBoot雪花算法主鍵ID傳到前端后精度丟失問題如何解決

可以看到,原來id為1352213368413982722,序列化為JS對象后變成了 1352213368413982700

代碼為:

const json = '{"id": 1352213368413982722, "name": "Tony"}';
const obj = JSON.parse(json);
 
console.log(obj.id);
console.log(obj.name);

解決方案

有如下兩種方案

  • 將數據庫表設計的id字段由 Long 類型改成 String 類型。

  • 前端用String類型來保存ID保持精度,后端及數據庫繼續使用Long(BigINT)類型

方案1使用String 類型做數據庫ID,查詢性能會大幅度下降。所以應該采用方案2。本文介紹方案2。

法1:全局處理

簡介

自定義ObjectMapper。

方案1:ToStringSerializer(推薦)

package com.knife.config;
 
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.databind.ser.std.ToStringSerializer;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
 
@Configuration
public class JacksonConfig {
 
    @Bean
    public ObjectMapper jacksonObjectMapper(Jackson2ObjectMapperBuilder builder) {
        ObjectMapper objectMapper = builder.createXmlMapper(false).build();
 
        // 全局配置序列化返回 JSON 處理
        SimpleModule simpleModule = new SimpleModule();
        // 將使用String來序列化Long類型
        simpleModule.addSerializer(Long.class, ToStringSerializer.instance);
        simpleModule.addSerializer(Long.TYPE, ToStringSerializer.instance);
        objectMapper.registerModule(simpleModule);
        return objectMapper;
    }
}

測試 

訪問:http://localhost:8080/user/find?id=1352213368413982722

SpringBoot雪花算法主鍵ID傳到前端后精度丟失問題如何解決

方案2:自定義序列化器(不推薦)

序列化器

package com.knife.config;
 
import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.databind.SerializerProvider;
import com.fasterxml.jackson.databind.annotation.JacksonStdImpl;
import com.fasterxml.jackson.databind.ser.std.NumberSerializer;
 
import java.io.IOException;
 
/**
 * 超出 JS 最大最小值 處理
 */
@JacksonStdImpl
public class BigNumberSerializer extends NumberSerializer {
 
	/**
	 * 根據 JS Number.MAX_SAFE_INTEGER 與 Number.MIN_SAFE_INTEGER 得來
	 */
	private static final long MAX_SAFE_INTEGER = 9007199254740991L;
	private static final long MIN_SAFE_INTEGER = -9007199254740991L;
 
	/**
	 * 提供實例
	 */
	public static final BigNumberSerializer instance = new BigNumberSerializer(Number.class);
 
	public BigNumberSerializer(Class<? extends Number> rawType) {
		super(rawType);
	}
 
	@Override
	public void serialize(Number value, JsonGenerator gen, SerializerProvider provider) throws IOException {
		// 超出范圍 序列化位字符串
		if (value.longValue() > MIN_SAFE_INTEGER && value.longValue() < MAX_SAFE_INTEGER) {
			super.serialize(value, gen, provider);
		} else {
			gen.writeString(value.toString());
		}
	}
}

ObjectMapper配置

package com.knife.config;
 
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.databind.ser.std.ToStringSerializer;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
 
@Configuration
public class JacksonConfig {
 
    @Bean
    public ObjectMapper jacksonObjectMapper(Jackson2ObjectMapperBuilder builder) {
        ObjectMapper objectMapper = builder.createXmlMapper(false).build();
 
        // 全局配置序列化返回 JSON 處理
        SimpleModule simpleModule = new SimpleModule();
        // 將使用自定義序列化器來序列化Long類型
        simpleModule.addSerializer(Long.class, BigNumberSerializer.instance);
        simpleModule.addSerializer(Long.TYPE, BigNumberSerializer.instance);
        objectMapper.registerModule(simpleModule);
        return objectMapper;
    }
}

測試 

訪問:http://localhost:8080/user/find?id=1352213368413982722

SpringBoot雪花算法主鍵ID傳到前端后精度丟失問題如何解決

法2:局部處理

說明

在字段上加:@JsonSerialize(using= ToStringSerializer.class)。

實例

package com.knife.entity;
 
import com.fasterxml.jackson.databind.annotation.JsonSerialize;
import com.fasterxml.jackson.databind.ser.std.ToStringSerializer;
import lombok.Data;
 
@Data
public class UserVO {
    @JsonSerialize(using= ToStringSerializer.class)
    private Long id;
 
    private String username;
}

測試 

訪問:http://localhost:8080/user/find?id=1352213368413982722

SpringBoot雪花算法主鍵ID傳到前端后精度丟失問題如何解決

到此,關于“SpringBoot雪花算法主鍵ID傳到前端后精度丟失問題如何解決”的學習就結束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學習,快去試試吧!若想繼續學習更多相關知識,請繼續關注億速云網站,小編會繼續努力為大家帶來更多實用的文章!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

闽清县| 三台县| 米脂县| 伽师县| 扶风县| 开江县| 巴彦县| 大新县| 武乡县| 恩平市| 奎屯市| 长治市| 洛宁县| 茂名市| 理塘县| 榆社县| 临江市| 淮滨县| 龙山县| 老河口市| 都匀市| 凤庆县| 元阳县| 琼结县| 亚东县| 灵川县| 潮安县| 辉县市| 方山县| 和田市| 仁布县| 阜阳市| 康平县| 宕昌县| 临湘市| 乐亭县| 四川省| 高尔夫| 京山县| 余庆县| 平陆县|