场景:
我们在设计数据库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,规定映射关系**
本文转载自: https://blog.csdn.net/qq_52692506/article/details/132065715
版权归原作者 稳健才是放肆 所有, 如有侵权,请联系我们删除。
版权归原作者 稳健才是放肆 所有, 如有侵权,请联系我们删除。