今天遇到一个问题,数据库数据无误,接口请求数据无误,但在浏览器发起请求后,接口数据在浏览器的
preview
和response
中显示不一致。数据库中的数据类型是bigint
,实体类对应的数据类型是Long
。问题表现如下:
JavaScript
的 Number
类型不能完全表示 Java
的 Long
类型数字。当 Long
类型的长度超过 17
位时,会出现精度丢失的问题。浏览器在解析超过 17
位的数字时,超出的部分会被转换为 0,导致显示不一致。具体原因如下:
JavaScript
中的 Number
类型是双精度浮点数,可以安全地表示的整数范围是 -9007199254740991
到 9007199254740991
(即 15
位数字)。当数字长度超过 15
位时,会发生精度丢失,超过 17
位的部分则会直接变成 0
。Java
的 Long
类型值为 1816022064764096513
,但在 JavaScript
中,超过 17
位的部分可能会变成 1816022064764096000
。为了避免精度丢失,可以在后台将 Long
型数据转换为 String
类型。这确保了在传输过程中数字不会丢失精度,并且前端可以正确地显示这些数据。
@Getter
@Setter
@ToString
@Schema(title = "[字典类型表参数对象]")
public class SysDictTypeDTO implements Serializable {
@Serial
private static final long serialVersionUID = 1L;
@Schema(title = "字典主键")
private String dictId; // 将 Long 修改为 String
}
通过这种方式,可以确保 Long
类型的数据在传输过程中不会丢失精度,前端可以正确显示大整数值。
新建全局配置类,处理 Jackson
导致 Long
型数据精度丢失问题
import com.fasterxml.jackson.annotation.JsonInclude;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.databind.ser.std.ToStringSerializer;
import lombok.extern.slf4j.Slf4j;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
/**
* Description: [解决 Jackson 导致 Long 型数据精度丢失问题]
* Author: [mobaijun]
* Date: [2024/8/1 15:58]
* IntelliJ IDEA Version: [IntelliJ IDEA 2023.1.4]
*/
@Slf4j
@Configuration
public class JacksonConfig {
/**
* 解决 Jackson 导致 Long 型数据精度丢失问题
*
* @return 返回 ObjectMapper 对象
*/
@Bean
public ObjectMapper jacksonObjectMapper(Jackson2ObjectMapperBuilder builder) {
ObjectMapper objectMapper = builder.createXmlMapper(false).build();
objectMapper.setSerializationInclusion(JsonInclude.Include.NON_NULL);
SimpleModule module = new SimpleModule();
module.addSerializer(Long.class, ToStringSerializer.instance);
module.addSerializer(Long.TYPE, ToStringSerializer.instance);
objectMapper.registerModule(module);
log.info("Jackson init: {}", objectMapper);
return objectMapper;
}
}
== 如果是多模块项目,启动类扫描这个包即可 ==