0


解决前端请求携带Long类型数据精度丢失的两种方案

场景:

     我们在设计数据库id的时候,会(使用MybatisPlus)使用**雪花算法生成的Long类型数据**作为主键id,但是当**前端发送请求携带的数据中有19位Long类型数据,他只能保证16位有效,最后3位会舍入,导致精度丢失。**

1.使用Springboot带的@JsonFormat,指定序列化和反序列化过程中要转换成的传输的类型

    比如我这里想让我的**Long类型的 id 在传输过程中使用String类型**,这样精度不会丢失,操作如下:

      同理,对于其他的类型,你也可以使用该注解指定,不过这样要一个一个加,比较繁琐,一不小心可能就漏了,下面有另一种方式

2.自定义对象映射器,实现全局的映射,管理全局的序列化和反序列化

①首先创建自定义的JacksonObjectMapper类(因为Jackson 是 Spring Boot 默认使用的对象映射器,通过配置 JacksonObjectMapper 类,可以定制 Jackson 的行为,以满足特定的序列化和反序列化需求。)

    代码如下:
package com.itheima.reggie.common;

import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.databind.ser.std.ToStringSerializer;
import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateDeserializer;
import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateTimeDeserializer;
import com.fasterxml.jackson.datatype.jsr310.deser.LocalTimeDeserializer;
import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateSerializer;
import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateTimeSerializer;
import com.fasterxml.jackson.datatype.jsr310.ser.LocalTimeSerializer;
import java.math.BigInteger;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.LocalTime;
import java.time.format.DateTimeFormatter;
import static com.fasterxml.jackson.databind.DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES;

/**
 * 对象映射器:基于jackson将Java对象转为json,或者将json转为Java对象
 * 将JSON解析为Java对象的过程称为 [从JSON反序列化Java对象]
 * 从Java对象生成JSON的过程称为 [序列化Java对象到JSON]
 *
 * 如果想生效,可以在增加配置类,然后封装成Bean
 */
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() {
        super();
        //收到未知属性时不报异常
        this.configure(FAIL_ON_UNKNOWN_PROPERTIES, false);

        //反序列化时,属性不存在的兼容处理
        this.getDeserializationConfig().withoutFeatures(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES);

        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);
    }
}

上述代码中,我想要实现的映射其实是这个部分(见图)

②写配置类封装成Bean

@Configuration
  public class JacksonConfig {
 
      @Bean
      public JacksonObjectMapper jacksonObjectMapper() {
          return new JacksonObjectMapper();
      }
  }

总结:

            对象映射可以参考这两种方法,**如果想一劳永逸可以采用方法2,如果只是处理偶尔几处,而可以自己手动加@JsonFormat,规定映射关系**
标签: java 开发语言

本文转载自: https://blog.csdn.net/qq_52692506/article/details/132065715
版权归原作者 稳健才是放肆 所有, 如有侵权,请联系我们删除。

“解决前端请求携带Long类型数据精度丢失的两种方案”的评论:

还没有评论