【关于Long类型的精度丢失问题】

前提:使用雪花算法生成ID作为数据库表的主键,导致雪花ID传到前端会出现精度丢失的问题,变现为前端显示ID和数据库中的ID不一致。
原因:Long类型的雪花ID有19位,而前端接收Long类型用的是number类型,但是number类型的精度只有16位。这就导致雪花ID传到前端会出现精度丢失。
解决方案:设置配置类,对SpringMVC中的一个消息转换器MappingJackson2HttpMessageConverter进行扩展。

import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Bean;
import com.fasterxml.jackson.annotation.JsonInclude;
import com.fasterxml.jackson.databind.MapperFeature;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.databind.ser.std.ToStringSerializer;
import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
import org.springframework.http.converter.json.MappingJackson2HttpMessageConverter;

/**
 * @description: Long 转为 String 防止丢失精度的配置类
 **/

@Configuration
public class JacksonConfig {
    @Bean
    public MappingJackson2HttpMessageConverter jackson2HttpMessageConverter() {
        final Jackson2ObjectMapperBuilder builder = new Jackson2ObjectMapperBuilder();
        builder.serializationInclusion(JsonInclude.Include.NON_NULL);
        final ObjectMapper objectMapper = builder.build();
        SimpleModule simpleModule = new SimpleModule();
        // Long 转为 String 防止 js 丢失精度
        simpleModule.addSerializer(Long.class, ToStringSerializer.instance);
        objectMapper.registerModule(simpleModule);
        // 忽略 transient 关键词属性
        objectMapper.configure(MapperFeature.PROPAGATE_TRANSIENT_MARKER, true);
        return new MappingJackson2HttpMessageConverter(objectMapper);
    }
}

Java 中,使用 `@JsonFormat` 注解可以有效防止 `long` 类型数据在序列化为 JSON 时出现精度丢失问题。通过将 `long` 类型的字段格式化为字符串形式,可以规避 JSONJavaScript 在处理大整数时的精度限制。具体实现方式如下: ### 使用 `@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` 类型字段的序列化行为。 ###
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值