Java开发 - 消失的id

目录

前言

问题描述

问题分析

解决方案

创建对象转换器

扩展原对象转换器

结语


前言

开发时,我们经常会遇到一些奇奇怪怪的问题,明明一切流程都是正确的,但却得不到正确的结果,这让我们很是苦恼,本系列博客将带你一起见识这些开发中存在的灵异事件,让我们一起解密他们,作为本系列的第一篇博客,我很荣幸的给大家介绍这篇消失的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<HttpMessageConverter<?>> converters) {
    log.info("扩展消息转换器...");
    //创建消息转换器对象
    MappingJackson2HttpMessageConverter messageConverter = new MappingJackson2HttpMessageConverter();
    //设置对象转换器,底层使用Jackson将Java对象转为json
    messageConverter.setObjectMapper(new JacksonObjectMapper());
    //将上面的消息转换器对象追加到mvc框架的转换器集合中
    converters.add(0,messageConverter);
}

一定要放在第一位,在程序里是第0位,这样才能让它生效。 

结语

到此,这个问题被解决了。这是开发中一个很小的问题,稍不留神就可能造成这种严重的影响,问题是还不好发现,新手往往会对此类问题束手无策,我们需要做的就是:只要时间的堆积和经验的积累,从一只菜鸟成为一只老鸟。开发没有捷径可走,如同人生,需要一步步坚定的走下去。我是CodingFire,你的良师益友,看到这里,很高兴能帮到你,咱们下篇再见。

猜你喜欢

转载自blog.csdn.net/CodingFire/article/details/129914415