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

目录

场景:

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

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

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

 ②写配置类封装成Bean

        总结:


场景:

         我们在设计数据库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,规定映射关系

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值