项目中使用雪花ID作为主键,雪花ID是19位Long类型数字,数据返回到前端会出现精度丢失问题,数字已经超过了前端浏览器或JS的最大值。
Java后端数据模型
返回到浏览器后的数据模型,前后数据不一致
序列化时将Long类型转成String类型
package com.olive.config;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.databind.ser.std.ToStringSerializer;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
import com.fasterxml.jackson.databind.ser.std.ToStringSerializer;
/**
* 全局序列化配置类
*/
@Configuration
public class JsonConfig {
/**
* 创建Jackson对象映射器
*
* @param builder Jackson对象映射器构建器
* @return
*/
@Bean
public ObjectMapper getJacksonObjectMapper(Jackson2ObjectMapperBuilder builder) {
ObjectMapper objectMapper = builder.createXmlMapper(false).build();
//序列换成json时,将所有的long变成string.因为js中得数字类型不能包含所有的java long值,超过16位后会出现精度丢失
SimpleModule simpleModule = new SimpleModule();
simpleModule.addSerializer(Long.class, ToStringSerializer.instance);
simpleModule.addSerializer(Long.TYPE, ToStringSerializer.instance);
objectMapper.registerModule(simpleModule);
return objectMapper;
}
}
import com.fasterxml.jackson.databind.annotation.JsonSerialize;
import com.fasterxml.jackson.databind.ser.std.ToStringSerializer;
import lombok.Data;
import lombok.ToString;
@Data
@ToString
public class UserDTO {
@JsonSerialize(using = ToStringSerializer.class)
private Long id;
@ToString.Exclude
private Long salary;
}
在前端使用JSONbig将大数字做安全处理
在package.json中引入JSONbig, "json-bigint": "^1.0.0"
,执行npm install命令
在requesst.js中创建axios实例时增加代码处理,导入JSONbig和添加transformResponse属性,相当于拦截器,请求相应后处理先处理一下。
//导入JSONbig
import JSONbig from 'json-bigint'
// 在创建axios实例中增加transformResponse属性
const service = axios.create({
// axios中请求配置有baseURL选项,表示请求URL公共部分
baseURL: process.env.VUE_APP_BASE_API,
transformResponse: [function(data) {
try {
// 作用1:把json字符串转为js对象
// 作用2:把里面的大数字做安全处理
return JSONbig.parse(data)
} catch (e) {
return data
}
}],
// 超时
timeout: 10000
})