全局处理Long类型字段返回前端精度丢失

在后端使用MybatisPlus时,主键策略为@TableId(type=IdType.ASSIGN_ID)生成19位的雪花ID,当以Long类型返回给前端时,由于number类型最多只能精确到16位,导致精度丢失。解决方法包括在字段上添加@JsonFormat(shape=JsonFormat.Shape.STRING)注解或全局配置Jackson序列化,确保Long类型转换为字符串以避免精度问题。

项目场景:

后端返回一个列表数据到前端,其中有部分字段类型后端定义的是Long类型并且是用雪花算法生成的,从响应数据中发现返回的值与数据库的不一致,丢失了精度


问题描述

后端使用MybatisPlus时,主键字段的主键策略用是@TableId(type = IdType.ASSIGN_ID),意味着主键会生成一个长度为19位的值,后端用Long类型接收没问题,而前端number能接收的最大长度为16位,最大值(9007 1992 5474 0992),此时后端雪花ID返回到前端就出现了丢失精度的情况。
如下图所示:

/**任务主键*/
@ApiModelProperty(value = "任务主键")
@TableId(type = IdType.ASSIGN_ID)
private Long taskId;

丢失精度后


解决方案:

有两种最常见的解决办法:
第一种是局部针对某个字段处理,这种是极其不推荐的,到处都得加;
第二种是全局配置,处理Long类型字段,推荐第二种

1、在字段上加注解 @JsonFormat(shape = JsonFormat.Shape.STRING)

/**任务主键*/
@ApiModelProperty(value = "任务主键")
@JsonFormat(shape = JsonFormat.Shape.STRING)
private Long taskId;

2、增加全局配置类,实现WebMvcConfigurer接口,重写configureMessageConverters方法

@Configuration
public class WebMvcConfiguration implements WebMvcConfigurer {

	/**
     * 添加Long转json精度丢失的配置
     * @Return: void
     */
    @Override
    public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
        MappingJackson2HttpMessageConverter jackson2HttpMessageConverter = new MappingJackson2HttpMessageConverter();
        ObjectMapper objectMapper = new ObjectMapper();
        SimpleModule simpleModule = new SimpleModule();
        simpleModule.addSerializer(Long.class, ToStringSerializer.instance);
        simpleModule.addSerializer(Long.TYPE, ToStringSerializer.instance);
        objectMapper.registerModule(simpleModule);
        jackson2HttpMessageConverter.setObjectMapper(objectMapper);
        converters.add(jackson2HttpMessageConverter);
    }
}

测试:

![](https://img-blog.csdnimg.cn/b946298690c94205af683f4d95c81d40.png

Java 中,使用 `@JsonFormat` 注解可以有效防止 `long` 类型数据在序列化为 JSON 时出现精度丢失问题。通过将 `long` 类型字段格式化为字符串形式,可以规避 JSON 和 JavaScript 在处理大整数时的精度限制。具体实现方式如下: ### 使用 `@JsonFormat` 注解 `@JsonFormat` 注解允许开发者指定字段的序列化格式。通过设置 `shape = Shape.STRING`,可以将 `long` 类型字段以字符串形式输出,从而避免精度丢失问题。例如: ```java import com.fasterxml.jackson.annotation.JsonFormat; import com.fasterxml.jackson.databind.annotation.JsonDeserialize; import com.fasterxml.jackson.databind.annotation.JsonSerialize; @JsonFormat(shape = JsonFormat.Shape.STRING) private Long id; ``` 这样,当字段 `id` 被序列化为 JSON 时,其值将被转换为字符串,确保在前端(如 JavaScript)中能够正确处理大整数[^2]。 ### 与 `@JsonSerialize` 的对比 除了 `@JsonFormat` 外,还可以使用 `@JsonSerialize` 注解来实现类似的功能。例如: ```java import com.fasterxml.jackson.databind.annotation.JsonSerialize; import com.fasterxml.jackson.databind.ser.std.ToStringSerializer; @JsonSerialize(using = ToStringSerializer.class) private Long id; ``` `@JsonSerialize` 提供了更细粒度的控制,允许指定特定的序列化器(如 `ToStringSerializer`)。与 `@JsonFormat` 相比,`@JsonSerialize` 更适用于需要自定义序列化逻辑的场景[^4]。 ### 全局配置替代方案 虽然 `@JsonFormat` 和 `@JsonSerialize` 提供了字段级别的解决方案,但在实际开发中,如果需要对所有 `long` 类型字段进行统一处理,建议采用全局配置方式。例如,在 Spring 应用中,可以通过自定义 `ObjectMapper` 来实现全局的 `long` 类型序列化配置: ```java import com.fasterxml.jackson.databind.ObjectMapper; import com.fasterxml.jackson.databind.ser.std.ToStringSerializer; import com.fasterxml.jackson.databind.module.SimpleModule; import org.springframework.context.annotation.Configuration; import org.springframework.http.converter.HttpMessageConverter; import org.springframework.http.converter.json.MappingJackson2HttpMessageConverter; import org.springframework.web.servlet.config.annotation.WebMvcConfigurer; import java.util.List; @Configuration public class WebConfig implements WebMvcConfigurer { @Override public void configureMessageConverters(List<HttpMessageConverter<?>> converters) { ObjectMapper objectMapper = new ObjectMapper(); SimpleModule module = new SimpleModule(); module.addSerializer(Long.class, ToStringSerializer.instance); module.addSerializer(Long.TYPE, ToStringSerializer.instance); objectMapper.registerModule(module); MappingJackson2HttpMessageConverter converter = new MappingJackson2HttpMessageConverter(objectMapper); converters.add(converter); } } ``` 通过这种方式,可以避免在每个字段上手动添加注解,提高代码的可维护性[^3]。 ### 总结 - `@JsonFormat(shape = Shape.STRING)` 可以将 `long` 类型字段以字符串形式输出,防止精度丢失。 - `@JsonSerialize(using = ToStringSerializer.class)` 提供了更灵活的序列化控制。 - 全局配置 `ObjectMapper` 可以统一处理所有 `long` 类型字段的序列化行为。 ###
评论 3
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值