热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

前后对接数字几_前后端超大整数传输会有精度损失?看看阿里怎么做

在阅读《阿里巴巴Java开发手册》时,发现有一条关于前后端超大整数返回的规约,这是JavaScript的一个坑:首先我们创建一个Sprin

在阅读《阿里巴巴Java开发手册》时,发现有一条关于前后端超大整数返回的规约,这是Javascript 的一个坑:

12a7096d5fee430b41304f80e2405d63.png

首先我们创建一个Spring Boot项目,然后再新建接口用来返回DbScript对象,其中的id是由mybatis-plus的IdWorker.getld生成的19位long类型的数值。

@RestController @RequestMapping("/dbScrip") public class DbScriptController { Logger logger = LoggerFactory.getLogger(DbScriptController.class); @RequestMapping("/info") public DbScript getDbScript() { DbScript dbScript = new DbScript(); // id long id = IdWorker.getId(); dbScript.setId(id); logger.info("id:{}", id); return dbScript; } }

然后将服务启动,通过浏览器对接口进行访问我们可以发现,通过日志可以看到后端传给前端的id是不同的,在这个传递的过程中产生了损失,后端传的是1304270071757017088但是前端接收的却是1304270071757017000。

我们通过开发手册来查询原因就会了解到,如果返回的数值大于2的53次方,那么就会转换成JS的Number,这些数值就可能在传输的过程中发生精度的损失。

解决方法

针对这项问题,可以采取如下的几种方法

如果这个对象只在这个方法中用到了,可以将该属性直接从Long变为String型。如果这个对象在很多地方都使用了,那么在序列化的过程中,将Long类型转换成String类型。我们也可以通过添加一个新的String属性来处理前后端对于大整数的传输。

方法一:

直接将Long id改成String id;这种方法相对来说比较局限,仅适用于对象只在这个方法中使用。

方法二:

还可以在属性上增加注解。这种方法也是有弊端存在的,如果需要修改的部分比较多,那么逐个添加的工作量还是比较大的,假设使用的是Jackson,那么可以通过WRITE_NUMBERS_AS_STRINGS这个配置参数将所有数字全部转换成字符串强制输出,方法使用也比较简单,优点在于使用方便,几乎不需要调整代码。但是这样做的问题在于颗粒度较大, 将所有的数字都转换成字符串输出后,按照 timestamp 格式输出的时间也会是这样。

方法三:

可以用一个多的属性,例如使用String dbScripId来代替之前的id。

总结

本文针对《阿里巴巴Java开发手册》中的对于需要使用超大整数的场景,服务端一律使用 String 字符串类型返回,禁止使用Long 类型出发,提出了几种解决方法,大家可以根据自己的需求去选择方法。



推荐阅读
author-avatar
手机用户2502933251
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有