雪花算法生成的ID在返回给前端之后和生成的不一样,到底是什么原因?

Posted 似水流年,是谁苍白了等待

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了雪花算法生成的ID在返回给前端之后和生成的不一样,到底是什么原因?相关的知识,希望对你有一定的参考价值。

一、前言

最近在做项目的时候发现用雪花算法生成的id传给前端以后跟生成的不一样,就纳闷,在想为什么会出现这样的问题?

二、问题描述:

雪花算法生成id为16位,返回到前端之后后两位变为0

从两个点出发:

1.第一次想到的是四舍五入,仔细检查了也不是四舍五入,有些比5大的数也变为0了。

2.查Long类型和String类型数据长度。

上网查了一下,long类型继承的是number类,而number类型精度为16位,而雪花算法生成的id为19位,因此会导致进度丢失

插入数据库和返回前端对比

所以初步判断这个问题肯定就是进度丢失,所以按着这个思路解决。

三、解决方案:

第一种解决方案

如果想要前端不丢失精度,JSON中的id就不能是long类型,改为String类型就好了。

这个方案如果此id用的类少,就可以改id的数据类型,但是当用到的类多的话要改过来就需要花很长时间,并且不能保证都能改过来,而且不会出问题。

第二种解决方案

通过添加一个全局配置来使long类型转为JSON中的string类型,省去了一个一个添加注解的麻烦。

 

@Configurationpublic class JacksonConfig 
  @Bean
  @Primary
  @ConditionalOnMissingBean(ObjectMapper.class)
  public ObjectMapper jacksonObjectMapper(Jackson2ObjectMapperBuilder builder)
  
    ObjectMapper objectMapper = builder.createXmlMapper(false).build();
    // 全局配置序列化返回 JSON 处理
    SimpleModule simpleModule = new SimpleModule();
    //JSON Long ==> String
    simpleModule.addSerializer(Long.class, ToStringSerializer.instance);
    objectMapper.registerModule(simpleModule);
    return objectMapper;
  

第三种解决方案

在application.yml中加上以下配置,这个办法会将所有数字都变成字符串,包括long和int类型

spring:
  jackson:
    generator:
      writeNumbersAsStrings: true

第四种解决方案:

在对应的字段上面添加注解

@JsonSerialize(using = ToStringSerializer.class)

我本次解决采用的是将id转为String类型,因为在项目中这个字段用到的少,容易修改,也不会影响其他功能。

以上是关于雪花算法生成的ID在返回给前端之后和生成的不一样,到底是什么原因?的主要内容,如果未能解决你的问题,请参考以下文章

雪花算法生成的ID,前端无法使用

ID号生成 雪花算法

解决雪花算法生成的ID传输前端后精度丢失

Oracle Id生成算法 —— 雪花算法

Oracle Id生成算法 —— 雪花算法

关于MyBatis-Plus雪花算法生成id精度丢失问题的处理