背景
记录一次令人蛋疼的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("/**");
}