数据库中有一个bigint类型数据,对应java中的类型为Long型,数据库中存储的是:1822916035657994240,显示出来却成了1822916035657994200,后面几位全变成了0造成精度丢失
原因:
JavaScript 的 Number 类型使用 64 位的双精度浮点数格式(IEEE 754 标准)。这 64 位的具体分配如下:
- 1 位用于符号位(正或负)
- 11 位用于指数部分,用来表示数值的规模(范围)
- 52 位用于存储有效数字(尾数),再加上一个隐含的最高有效位
尾数位最大是 52 位,因此 JS 中能精准表示的最大整数是 Math.pow(2, 53),十进制即 9007199254740992。而Bigint类型的有效位数是63位(扣除一位符号位),其最大值为:Math.pow(2,63)。序列化之后的数值大于了 9007199254740992 的就可能会丢失精度
解决办法
@Configuration
public class JacksonConfig {
@Bean
@Primary
@ConditionalOnMissingBean(ObjectMapper.class)
public ObjectMapper jacksonObjectMapper(Jackson2ObjectMapperBuilder builder) {
ObjectMapper objectMapper = builder.createXmlMapper(false).build();
SimpleModule simpleModule = new SimpleModule();
//JSON Long ==> String
simpleModule.addSerializer(Long.class, ToStringSerializer.instance);
objectMapper.registerModule(simpleModule);
return objectMapper;
}
}
jackson序列化Long的时候指定一下规则即可,序列化成String,这样就不用修改Long类型为String。
结论
默认序列化为数值的时候精度丢失,js中紫色表示数值
修改序列化规则Long -> String,js中橙色是字符串