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

Posted 白日日白

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了解决雪花算法生成的ID传输前端后精度丢失相关的知识,希望对你有一定的参考价值。

本章目录:

  • 问题描述
  • 解决方案
    • 修改数据库字段
    • 配置MVC全局消息转换器
    • 修改Result类

 一、问题描述 

在用雪花算法生成的ID传输到前端时,会出现后三位精度丢失

可以看到,我们得到的response为1594605819398193154

而前端展示的为1594605819398193200

这是因为JS是弱语言,前端接收数字类型参数为number

最大接收长度为16位,超出长度则会丢失精度

而JAVA的Long类型长度为19位,所以传输到前端的后三位精度丢失

所以我们解决该问题的思路就是:把java里的Long类型转换为String返回给前端

 

二、解决方案

2.1、修改数据库字段

可以把数据库的ID字段从Long改为varchar,不过不建议这样做,这样修改后查询无法走索引,降低了效率

 

2.2、配置MVC全局消息转换器

我们可以在springmvc的配置类中(也就是继承了WebMvcConfigurationSupport接口的那个配置类)重写extendMessageConverters方法

 

    /**
     * 扩展mvc框架的消息转换器
     * @param converters
     */
    @SneakyThrows
    @Override
    protected void extendMessageConverters(List<HttpMessageConverter<?>> converters) 
        //创建消息转换器对象
        MappingJackson2HttpMessageConverter messageConverter=new MappingJackson2HttpMessageConverter();
        //设置对象转换器,底层使用Jackson将Java对象转为json
        messageConverter.setObjectMapper(new JacksonObjectMapper());
        //将上面的消息转换器对象追加到mvc框架的转换器集合中
        converters.add(0,messageConverter);
    

编写一个JacksonObjectMapper类

/**
 * 对象映射器:基于jackson将Java对象转为json,或者将json转为Java对象
 * 将JSON解析为Java对象的过程称为 [从JSON反序列化Java对象]
 * 从Java对象生成JSON的过程称为 [序列化Java对象到JSON]
 */
public class JacksonObjectMapper extends ObjectMapper 

    public static final String DEFAULT_DATE_FORMAT = "yyyy-MM-dd";
    public static final String DEFAULT_DATE_TIME_FORMAT = "yyyy-MM-dd HH:mm:ss";
    public static final String DEFAULT_TIME_FORMAT = "HH:mm:ss";

    public JacksonObjectMapper() throws NoSuchFieldException 
        SimpleModule simpleModule = new SimpleModule()
                .addDeserializer(LocalDateTime.class, new LocalDateTimeDeserializer(DateTimeFormatter.ofPattern(DEFAULT_DATE_TIME_FORMAT)))
                .addDeserializer(LocalDate.class, new LocalDateDeserializer(DateTimeFormatter.ofPattern(DEFAULT_DATE_FORMAT)))
                .addDeserializer(LocalTime.class, new LocalTimeDeserializer(DateTimeFormatter.ofPattern(DEFAULT_TIME_FORMAT)))

                .addSerializer(BigInteger.class, ToStringSerializer.instance)
                .addSerializer(Long.class, ToStringSerializer.instance)
                .addSerializer(LocalDateTime.class, new LocalDateTimeSerializer(DateTimeFormatter.ofPattern(DEFAULT_DATE_TIME_FORMAT)))
                .addSerializer(LocalDate.class, new LocalDateSerializer(DateTimeFormatter.ofPattern(DEFAULT_DATE_FORMAT)))
                .addSerializer(LocalTime.class, new LocalTimeSerializer(DateTimeFormatter.ofPattern(DEFAULT_TIME_FORMAT)));

        //注册功能模块 例如,可以添加自定义序列化器和反序列化器
        this.registerModule(simpleModule);
    

注意:该方法是将所有Long类型都序列化成String的《全局转换》 

 

2.3、修改Result类型

我这边封装了一个Result类,可以直接用String.valueOf去把Long类型的id转换成String

也可以在Result类里多加一个判断

 

 

 

 

 

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

MyBatis-Plus3.3.0开始,默认使用雪花算法

首先插入一条数据 生成的id长这样

1400693801379520513

但是我们传给前端的时候却是这样

1400693801379520500

上网查了一下Number精度是16位(雪花ID是19位的)
So:前端的Number数据类型导致的精度丢失。

解决问题

这时候 我们只需在 private Long id 上面加个注解即可 无需其他操作

 @JsonFormat(shape = JsonFormat.Shape.STRING)
 private Long id;

@JsonFormat 用来表示json序列化的一种格式或者类型,shap表示序列化后的一种类型

以上是关于解决雪花算法生成的ID传输前端后精度丢失的主要内容,如果未能解决你的问题,请参考以下文章

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

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

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

Springboot解决雪花算法ID到前端精度丢失

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

雪花算法踩坑 - Long 类型 id 返回前端精度丢失 (通过序列化解决)