问题描述:
本人最近在写一个前后端分离项目,后端实体类中的id字段使用Long类型,来对应数据库bigint类型的主键。同时使用雪花算法自动生成17位数字的id。 (注:雪花算法,生成指定位数不重复的Long值,来作为主键的id) 然后前端获取id时,与后端传入的id不一致。 如下图:(后端返回的id) 前台实际接收的id如下图: 很明显,前台获取的id与后台数据库里查询出来的id不一致。
原因分析:
分析:查阅相关资料,发现前端中js的Number类型最大值为9007199254740991(16位数),一旦超过这个值,就会发生精度缺失现象。本文中的雪花算法产生的id为92625918782214144(17位数),明显大于Number类型的最大值,因此前后端数据传递发生错误,导致前端接收到的与后台传入的不一致。
注:js中Number类型最大值:9007199254740991(16位数) java中Long类型最大值:9223372036854775807(19位数) 本项目中id的值均为 17位数,因此符合Long类型,不符合Number类型。 因此本项目中的id会产生四舍五入的现象。 我的项目中 我真实的id为 92625918782214144 前端接收的为 92625918782214140
解决方案:
解决方案1(亲测有效)
Springboot项目,在配置文件中添加一句即可。
spring.jackson.generator.write_numbers_as_strings = true
后端还是使用Long类型,前端使用Number接收。
解决方案2
实体类中,id字段不使用Long类型,改为String类型即可。
private String id;
不推荐,这样的话你的数据库主键就不能使用bigint类型了,得使用varchar类型,varchar类型作为主键的话加大消耗,降低了索引效率。
解决方案3
使用alibaba的fastJSON工具包(需要引入依赖) 实体类中,在Long类型的字段上加@JsonSerialize(using = ToStringSerializer.class) 这样就能在序列化时自动将该字段类型转为String类型返给前端
@Data
public class Ebook {
@JSONField(serializeUsing= ToStringSerializer.class)
private Long id;
private String name;
或者 不要设置超过16位的Long类型就OK啦。
总结
目前就找到这么多,工作之余记录一下自己遇到的坑。如有不足,欢迎指正。
|