雪花算法ID到前端后产生精度丢失问题?
一、问题描述
数据库:
CREATE TABLE student
(
id BIGINT(20) NOT NULL COMMENT '主键ID',
......
);
实体类:
@Data
@EqualsAndHashCode(callSuper = false)
@Accessors(chain = true)
@TableName("student")
public class Student {
@TableId(type = IdType.ASSIGN_ID)
private Long id;
private String name;
private String phone;
}
由此产生的问题:
数据库中使用了
BIGINT
类型的主键ID,后端使用了MybatisPlus
中的雪花算法自动生成19位
长度的纯数字作为主键ID,但前端在接收19位
长度的主键ID时,由于JS
的数值有限,并不能正确接收,会产生精度问题,导致前端获取到的主键ID与数据库中的主键ID不一致,无法进行后续的操作。
二、解决办法
解决思路:
后端主键ID(Long) ==> Jackson(Long转String) ==> 前端使用String类型的ID,前端使用String类型的ID,就不会产生精度问题。
那么前端将String类型的19位数字传递给服务端时,服务端可以用Long类型接收吗?当然可以,因为这是Spring反序列化参数接收默认支持的行为。
2.1 局部设置
@Data
@AllArgsConstructor
class Student{
// 表示该字段,以字符串类型,返回给前端 1101101101101101101 => "1101101101101101101"
@JsonSerialize(using = ToStringSerializer.class)
private Long id;
private String name;
private String phone;
}
{
"id": "1101101101101101101",
"name": "张三",
"phone": "18179676724"
}
2.2 全局设置
@Configuration
@EnableWebMvc
public class WebMvcConfig implements WebMvcConfigurer {
@Override
public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
MappingJackson2HttpMessageConverter jackson2HttpMessageConverter = new MappingJackson2HttpMessageConverter();
ObjectMapper objectMapper = new ObjectMapper();
// 全局配置序列化返回 JSON 处理
SimpleModule simpleModule = new SimpleModule();
// JSON Long ==> String 将所有的 Long类型 转换成 String类型 返回
simpleModule.addSerializer(Long.class, ToStringSerializer.instance);
objectMapper.registerModule(simpleModule);
jackson2HttpMessageConverter.setObjectMapper(objectMapper);
converters.add(jackson2HttpMessageConverter);
}
}
{
"id": "1101101101101101101",
"name": "张三",
"phone": "18179676724"
}