乐趣区

使用JsonSerialize统一处理BigDecimal后缀0

1. 问题

    项目中 oracle 数据库需要转换为 mysql,Oracle 中的表字段定义为 number(36,16) 类型的工具自动转换为 mysql 的 decimal(36,16)。在 Oracle 数据库中,number(36,16) 类型的字段,使用 BigDecimal 类型进行映射字段,能够正常获取到去除后缀 0 的值。MySQL 数据库中存储为 decimal(36,16) 对应的字段,使用 BigDecimal 进行映射,会得到加后缀 0 的数值。例如在 oracle 中实际精度是 16 位的数值 0.000345344,使用 BigDecimal 接收,会得到 0.000345344,而 mysql 会得到的值为 0.0003453440000000,这样返回给前端,就会有问题。

2. 解决方法

  • 按照业务需求,在 mysql 数据库下,修改每一个字段对应的精度。然后在代码中手动对 BigDecimal 类型的字段进行手动去 0 后缀处理。
  • 在 vo 层进行 json 格式化中,进行去除 0 后缀。(这种方案不推荐,但要是想快速修改,可以采取)

      package com.xiayu.config;
    
      import com.fasterxml.jackson.core.JsonGenerator;
      import com.fasterxml.jackson.databind.JsonSerializer;
      import com.fasterxml.jackson.databind.SerializerProvider;
    
      import java.io.IOException;
      import java.math.BigDecimal;
      import java.util.Objects;
    
      public class CustomerBigDecimalSerialize extends JsonSerializer<BigDecimal> {
          @Override
          public void serialize(BigDecimal value, JsonGenerator gen, SerializerProvider serializers) throws IOException {if(Objects.nonNull(value)) {gen.writeNumber(value.stripTrailingZeros());
                  // 去除 0 后缀, 如果想统一进行保留精度,也可以采用类似处理
              }else {// 如果为 null 的话,就写 null
                  gen.writeNull();}
          }
      }
      
      // 在 vo 类字段中对字段加上 @JsonSerialize 注解
      @JsonSerialize(using = CustomerBigDecimalSerialize.class)
      private BigDecimal totalExchangeBtc; // 抵扣电费的 usdt 兑换的 btc 总量
      @JsonSerialize(using = CustomerBigDecimalSerialize.class)
      private BigDecimal perTotalExchangeBtc; // 兑换的 btc 总量对应每份的 btc
      @JsonSerialize(using = CustomerBigDecimalSerialize.class)
      private BigDecimal exchangeUsdtPrice; // 兑换时 usdt 的价格 
退出移动版