接口返回 preview 和 response 不一致 全局序列long转string

背景

记录一次令人蛋疼的bug排查。前段时间拿别人开源的博客项目搭建了自己的网站,为了安全,把用户id由自增改成了雪花算法生成。在调用支付时接口没有报错(后续有空打算出一篇个人如何对接微信支付的文章),但是金币未到账。
在这里插入图片描述
在这里插入图片描述

排查过程

一开始先去检查代码,并没有发现代码有异常,然后就去查看下日志,发现用户id不对,根本没有这个用户id。那么这个用户id到底是怎么来的呢?然后又去看下前端代码,有没有写死用户id,前端代码也没有任何问题呀。
最后,经过半个小时的摸索,偶然间发现接口返回 preview 和 respons不一致。后面根据经验的判断和网站的搜索,确定了是前端Number类型不能够显示那么大的数值。
(JavaScript中Number类型并不能完全表示Long型的数字,在Long长度大于17位时会出现精度丢失的问题,超过17位的部分浏览器会转换为0显示)

解决方案

在字段上使用@JsonSerialize(不建议,反正都要加代码了,为啥不一步到位,直接使用全局统一处理呢)

@JsonSerialize(using = ToStringSerializer.class)

通过重写WebMvcConfigurationSupport里的消息转换器方法实现(全局)


@Configuration
public class AllConfig extends WebMvcConfigurationSupport {

	/**
     * 扩展消息转换器
     *  目的: 主要是告诉jackson 转化JavaBean对象成json字符串的时候,具体的一些类型要转化成什么类型
     *     比如:  long 类型  ---> String类型
     * @param converters
     */
    @Override
    protected void extendMessageConverters(List<HttpMessageConverter<?>> converters) {
        //1. 创建自己的映射消息转换器 jackson 转化JavaBean成json字符串的转换器对象
        MappingJackson2HttpMessageConverter converter = new MappingJackson2HttpMessageConverter();
        //2. 把我们定义好的映射规则对象,丢到上面的转换器里面
        converter.setObjectMapper(new JacksonObjectMapper());
        //3. 把我们定义好的这个消息转换器,丢到集合里面去
        converters.add(0,converter);
    }
}

自定义的规则

/**
 * 对象映射器:基于jackson将Java对象转为json,或者将json转为Java对象
 * 将JSON解析为Java对象的过程称为 [从JSON反序列化Java对象]
 * 从Java对象生成JSON的过程称为 [序列化Java对象到JSON]
 * 
 * 该自定义的对象转换器, 主要指定了, 在进行json数据序列化及反序列化时, LocalDateTime、LocalDate、LocalTime的处理方式, 以及BigInteger及Long类型数据,直接转换为字符串。
 */
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);
    }
}

后续代码补充

一天后发现分页拦截器失效了。在AllConfig中重写addInterceptors()

添加代码:

    @Override
    public void addInterceptors(InterceptorRegistry interceptor) {
        interceptor.addInterceptor(new PageableInterceptor()).addPathPatterns("/**");

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值