目录
前言
问题描述
问题分析
解决方案
创建对象转换器
扩展原对象转换器
结语
开发时,我们经常会遇到一些奇奇怪怪的问题,明明一切流程都是正确的,但却得不到正确的结果,这让我们很是苦恼,本系列博客将带你一起见识这些开发中存在的灵异事件,让我们一起解密他们,作为本系列的第一篇博客,我很荣幸的给大家介绍这篇消失的id。
由于部分需求下,我们采用了雪花算法等算法自动生成id,这也就导致了id很长,通过mvc框架的转换器序列化后返回客户端后,当客户端拿着我们返回的id去获取信息的时候竟然提示数据库没有此id,真是见鬼了?
首先,我们通过查看控制台输出SQL发现,用户传入的id的确和数据库中的id有差异。由于此情况发生在分页查询时,我们来想想分页查询会发生什么:
服务端将返回给客户端的对象进行json序列化,将数据转换为json格式的数据,而我们实体类中的id是一个Long类型的数据,由于使用了雪花算法生成id,id的长度都是19位。
逻辑流程都没错,这好像没什么不妥啊?那问题究竟出在哪里呢?
有没有一种可能,问题不一定出在后端这里,那我们不妨从移动端入手找找原因?
经过我们通过对一个19位数字的普通输出发现:js在对长度较长的长整型数据进行处理时,会损失精度, 从而导致提交到后端的id和数据库中的id不一致。
原来如此,也就是在移动端接收到数据的那一刻起,也许是在编译器的作用下,这种精度的损失就开始了,有没有一种方式,可以使数据在传递的过程中不会因为数据的转化导致精度缺失呢?
别说,还真有:字符串!果然是万能的字符串啊,所以下一步,我们就需要让数据在传给前端的时候,id这类长整型数据可以转化为String类型。那有没有这种方法可以快捷方便的实现呢?
要想知道有没有此方法,我们还是要从转化本身着手,由于在SpringMVC中, 将Controller方法的返回值转换为json是通过jackson来实现的,它内部涉及到SpringMVC中的一个消息转换器:MappingJackson2HttpMessageConverter,所以我们可以从这个转换器入手,看看能不能对这个转换器进行扩展。
我们很幸运,SpringMVC允许我们对这个转换器进行扩展!!!
要想扩展,我们首先需要创建一个新的转换器来替换原来的转换器。
package com.codingfire.config;
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]
*/
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);
}
}
从代码中,我们看到除了对长整型数据进行转化外,还有对BigInteger,LocalDate,LocalTime等类型数据的转化。
下一步,我们需要让我们新写的转换器工作,我们需要重写一个叫extendMessageConverters的方法,该方法出现在WebMvcConfig类中,此类是一个自定义的配置类,它继承了WebMvcConfigurationSupport类,所以,我们在WebMvcConfig类中重写extendMessageConverters方法即可。
/**
* 扩展mvc框架的消息转换器
* @param converters
*/
@Override
protected void extendMessageConverters(List> converters) {
log.info("扩展消息转换器...");
//创建消息转换器对象
MappingJackson2HttpMessageConverter messageConverter = new MappingJackson2HttpMessageConverter();
//设置对象转换器,底层使用Jackson将Java对象转为json
messageConverter.setObjectMapper(new JacksonObjectMapper());
//将上面的消息转换器对象追加到mvc框架的转换器集合中
converters.add(0,messageConverter);
}
一定要放在第一位,在程序里是第0位,这样才能让它生效。
到此,这个问题被解决了。这是开发中一个很小的问题,稍不留神就可能造成这种严重的影响,问题是还不好发现,新手往往会对此类问题束手无策,我们需要做的就是:只要时间的堆积和经验的积累,从一只菜鸟成为一只老鸟。开发没有捷径可走,如同人生,需要一步步坚定的走下去。我是CodingFire,你的良师益友,看到这里,很高兴能帮到你,咱们下篇再见。