解决方式一:自定义全局序列化
在项目上线以后商品购买出现故障,原因是价格不对 30.495,但是数据库是30.50。
- 定义问题为BigDecimal失去精度
- 将BigDecimal类型的数据序列化成String类型传给前端解决问题。前端取值let 或者var 不是强类型。
一、自定义序列化类继承StdSerializer类重新serialize方法
public class BigDecimalStringSerializer extends StdSerializer<BigDecimal> {
public final static BigDecimalStringSerializer instance = new BigDecimalStringSerializer();
public BigDecimalStringSerializer() {
super(BigDecimal.class);
}
@Override
public void serialize(BigDecimal value, JsonGenerator gen, SerializerProvider provider) throws IOException {
if(value==null){
gen.writeString("0");
}else{
String val=value.stripTrailingZeros().toPlainString();
gen.writeString(val);
}
}
}
二、配置全局序列化
/**
* Jackson配置类
*/
@Configuration
public class JacksonConfig {
@Bean
@Primary
@ConditionalOnMissingBean(ObjectMapper.class)
public ObjectMapper jacksonObjectMapper(Jackson2ObjectMapperBuilder builder) {
ObjectMapper objectMapper = builder.createXmlMapper(false).build();
SimpleModule bigDecimalModule = new SimpleModule();
//序列化将BigDecimal转String类型
bigDecimalModule.addSerializer(BigDecimal.class, BigDecimalStringSerializer.instance);
bigDecimalModule.addKeySerializer(BigDecimal.class, BigDecimalStringSerializer.instance);
// 注册转换器
objectMapper.registerModule(bigDecimalModule);
// json不返回null的字段
objectMapper.setSerializationInclusion(JsonInclude.Include.NON_NULL);
return objectMapper;
}
}
三、测试
创建实体类用于测试
@Data
public class UserDetail {
private BigDecimal price;
private String phone;
}
@GetMapping("/user/detail")
public UserDetail userdetail(){
return new UserDetail(BigDecimal.valueOf(2L),"12345");
}
- 这样配置以后就是String类型了
解决方式二:加上注解@JsonSerialize(using= ToStringSerializer.class)
直接在BBigDecimal类型的字段加上注解@JsonSerialize(using= ToStringSerializer.class)
@JsonSerialize(using= ToStringSerializer.class)
private BigDecimal price;
- 作用效果一样,但是缺点是每个BigDecimal类型属性都需要加这个注解。