Long类型返回前端精度丢失

  • Post author:
  • Post category:其他




【1】给前端返回Long会出现精度丢失问题

在《阿里巴巴Java开发手册》中,有一条关于前后端超大整数返回的规约,具体内容如下:

在这里插入图片描述



【2】问题复现

后端直接用postman测试接口,返回数据没有问题。但是前端访问接口的时候,发现Long类型的最后两位被替换成了0,导致出现错误

请添加图片描述
为什么会发生这样的情况呢?

如果返回的数值超过 2 的 53 次方,就会转换成 JS 的 Number,此时有些数值就有可能发生精度损失。



【3】解决方法

(1)如果这个对象只在这个方法中用到了,可以将该属性直接从 Long 类型改为 String 类型。

(2)如果这个对象在很多地方都用到了,可以在序列化的时候,将 Long 类型转换成 String 类型。

(3)还可以添加一个新的 String 类型的属性,专门用来在前后端传输这种大整数。



(1)返回的信息把Long改成String

第一种方法比较简单,直接将 Long id; 改为 String id;,这种只适用于这个对象只在这个方法中使用了,比较局限。



(2)属性上增加注解@JsonFormat

第二种方法可以在属性上增加注解,如果使用的Jackson,可以添加 @JsonFormat(shape = JsonFormat.Shape.STRING) 或者 @JsonSerialize(using = ToStringSerializer.class) 注解。

如果这种需要修改的情况比较多,那么逐个添加还是有点费事,那么还有什么好办法吗?

如果使用的是Jackson,它有个配置参数 WRITE_NUMBERS_AS_STRINGS,可以强制将所有数字全部转成字符串输出,使用方法很简单,只需要配置参数即可:spring.jackson.generator.write_numbers_as_strings=true,这种方式的优点是使用方便,不需要调整代码;缺点是颗粒度太大,所有的数字都被转成字符串输出了,包括按照 timestamp 格式输出的时间也是如此。

那么还有什么方法能够只对 Long 类型进行处理转换成 String 类型呢?

Jackson 提供了这种支持,可以对 ObjectMapper 进行定制,具体代码如下所示:

public class JacksonConfiguration {

    @Bean

    public Jackson2ObjectMapperBuilderCustomizer jackson2ObjectMapperBuilderCustomizer() {

        return jacksonObjectMapperBuilder -> jacksonObjectMapperBuilder

                .serializerByType(Long.class, ToStringSerializer.instance)

                .serializerByType(Long.TYPE, ToStringSerializer.instance);

    }

}

通过定义 Jackson2ObjectMapperBuilderCustomizer,对 Jackson2ObjectMapperBuilder 对象进行定制,对 Long 型数据进行了定制,使用ToStringSerializer来进行序列化。



(3)添加冗余属性

第三种方法就需要多一个属性,比如使用String dbScripId,用来代替之前的 id。



版权声明:本文为weixin_44823875原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。