BigDecimal的操作工具类

import java.math.BigDecimal;
 
/**
 * 进行BigDecimal对象的加减乘除,四舍五入等运算的工具类 
 * @author ameyume
 *
 */
public class Arith {
 
    /**
     * 由于Java的简单类型不能够精确的对浮点数进行运算,这个工具类提供精  
     * 确的浮点数运算,包括加减乘除和四舍五入。  
     */
    //默认除法运算精度   
    private static final int DEF_DIV_SCALE = 10;
 
    //这个类不能实例化   
    private Arith(){
    }
 
    /**
     * 提供精确的加法运算。  
     * @param v1 被加数  
     * @param v2 加数  
     * @return 两个参数的和
     */
    public static double add(double v1,double v2){
        BigDecimal b1 = new BigDecimal(Double.toString(v1));
        BigDecimal b2 = new BigDecimal(Double.toString(v2));
        return b1.add(b2).doubleValue();
    }
 
    /**
     * 提供精确的减法运算。  
     * @param v1 被减数  
     * @param v2 减数  
     * @return 两个参数的差
     */
    public static double sub(double v1,double v2){
        BigDecimal b1 = new BigDecimal(Double.toString(v1));
        BigDecimal b2 = new BigDecimal(Double.toString(v2));
        return b1.subtract(b2).doubleValue();
    }
 
    /**
     * 提供精确的乘法运算。  
     * @param v1 被乘数  
     * @param v2 乘数  
     * @return 两个参数的积
     */
    public static double mul(double v1,double v2){
        BigDecimal b1 = new BigDecimal(Double.toString(v1));
        BigDecimal b2 = new BigDecimal(Double.toString(v2));
        return b1.multiply(b2).doubleValue();
    }
 
    /**
     * 提供(相对)精确的除法运算,当发生除不尽的情况时,精确到  
     * 小数点以后10位,以后的数字四舍五入。  
     * @param v1 被除数  
     * @param v2 除数  
     * @return 两个参数的商
     */
    public static double div(double v1,double v2){
        return div(v1,v2,DEF_DIV_SCALE);
    }
 
    /**
     * 提供(相对)精确的除法运算。当发生除不尽的情况时,由scale参数指  
     * 定精度,以后的数字四舍五入。  
     * @param v1 被除数  
     * @param v2 除数  
     * @param scale 表示表示需要精确到小数点以后几位。  
     * @return 两个参数的商
     */
    public static double div(double v1,double v2,int scale){
        if(scale<0){
            throw new IllegalArgumentException(
                    "The scale must be a positive integer or zero");
        }
        BigDecimal b1 = new BigDecimal(Double.toString(v1));
        BigDecimal b2 = new BigDecimal(Double.toString(v2));
        return b1.divide(b2,scale,BigDecimal.ROUND_HALF_UP).doubleValue();
    }
 
    /**
     * 提供精确的小数位四舍五入处理。  
     * @param v 需要四舍五入的数字  
     * @param scale 小数点后保留几位  
     * @return 四舍五入后的结果
     */
    public static double round(double v,int scale){
        if(scale<0){
            throw new IllegalArgumentException(
                    "The scale must be a positive integer or zero");
        }
        BigDecimal b = new BigDecimal(Double.toString(v));
        BigDecimal one = new BigDecimal("1");
        return b.divide(one,scale,BigDecimal.ROUND_HALF_UP).doubleValue();
    }
 
    /**
     * 提供精确的类型转换(Float)  
     * @param v 需要被转换的数字  
     * @return 返回转换结果
     */
    public static float convertsToFloat(double v){
        BigDecimal b = new BigDecimal(v);
        return b.floatValue();
    }
 
    /**
     * 提供精确的类型转换(Int)不进行四舍五入  
     * @param v 需要被转换的数字  
     * @return 返回转换结果
     */
    public static int convertsToInt(double v){
        BigDecimal b = new BigDecimal(v);
        return b.intValue();
    }
 
    /**
     * 提供精确的类型转换(Long)  
     * @param v 需要被转换的数字  
     * @return 返回转换结果
     */
    public static long convertsToLong(double v){
        BigDecimal b = new BigDecimal(v);
        return b.longValue();
    }
 
    /**
     * 返回两个数中大的一个值  
     * @param v1 需要被对比的第一个数  
     * @param v2 需要被对比的第二个数  
     * @return 返回两个数中大的一个值
     */
    public static double returnMax(double v1,double v2){
        BigDecimal b1 = new BigDecimal(v1);
        BigDecimal b2 = new BigDecimal(v2);
        return b1.max(b2).doubleValue();
    }
 
    /**
     * 返回两个数中小的一个值  
     * @param v1 需要被对比的第一个数  
     * @param v2 需要被对比的第二个数  
     * @return 返回两个数中小的一个值
     */
    public static double returnMin(double v1,double v2){
        BigDecimal b1 = new BigDecimal(v1);
        BigDecimal b2 = new BigDecimal(v2);
        return b1.min(b2).doubleValue();
    }
 
    /**
     * 精确对比两个数字  
     * @param v1 需要被对比的第一个数  
     * @param v2 需要被对比的第二个数  
     * @return 如果两个数一样则返回0,如果第一个数比第二个数大则返回1,反之返回-1  
     */
    public static int compareTo(double v1,double v2){
        BigDecimal b1 = new BigDecimal(v1);
        BigDecimal b2 = new BigDecimal(v2);
        return b1.compareTo(b2);
    }
 
} 

 

### 关于Spring Boot项目中使用BigDecimal工具类 在开发财务应用或其他需要高精度数值运算的应用时,`BigDecimal` 是 Java 中用于处理精确小数的最佳选择。为了简化 `BigDecimal` 的操作,在 Spring Boot 项目中通常会创建一些辅助工具类来封装常见的算术逻辑。 #### 封装BigDecimal比较的工具类 通过定义静态方法可以方便地对比两个 `BigDecimal` 对象之间的大小关系: ```java public class BigDecimalUtil { private BigDecimalUtil() {} public static int compare(BigDecimal a, BigDecimal b) { if (a == null || b == null) throw new IllegalArgumentException("Arguments cannot be null"); return a.compareTo(b); } } ``` 此段代码实现了基本的大于、小于等于判断功能[^1]。 #### 封装计算的工具类(支持加减乘除) 对于四则运算来说,则可以通过如下方式进一步扩展工具类的功能: ```java public final class BigDecimalMath { private static final MathContext MC = new MathContext(10); private BigDecimalMath() {} /** * 加法运算. */ public static BigDecimal add(BigDecimal... values) { return Arrays.stream(values).reduce(BigDecimal.ZERO, BigDecimal::add).round(MC); } /** * 减法运算. */ public static BigDecimal subtract(BigDecimal minuend, BigDecimal subtrahend) { Objects.requireNonNull(minuend, "Minuend must not be null."); Objects.requireNonNull(subtrahend, "Subtrahend must not be null."); return minuend.subtract(subtrahend).round(MC); } /** * 乘法运算. */ public static BigDecimal multiply(BigDecimal multiplicand, BigDecimal multiplier) { Objects.requireNonNull(multiplicand, "Multiplicand must not be null."); Objects.requireNonNull(multiplier, "Multiplier must not be null."); return multiplicand.multiply(multiplier).round(MC); } /** * 除法运算. */ public static BigDecimal divide(BigDecimal dividend, BigDecimal divisor, RoundingMode roundingMode) throws ArithmeticException { if (divisor.signum() == 0) { throw new ArithmeticException("Divisor can't be zero!"); } return dividend.divide(divisor, MC.getPrecision(), roundingMode); } } ``` 上述实现不仅考虑到了异常情况下的错误提示,还利用了 `RoundingMode` 来指定舍入模式以应对可能出现的小数位过多的问题。 #### 使用场景举例——账单管理系统的金额计算 假设有一个简单的账单管理系统,其中涉及到商品价格相加求总和的操作,此时就可以调用上面提到过的 `BigDecimalMath.add()` 方法来进行安全可靠的累加操作: ```java @Service public class BillService { @Autowired private ProductRepository productRepo; public BigDecimal calculateTotalAmount(List<Long> productIdList) { List<ProductEntity> products = productRepo.findAllById(productIdList); return BigDecimalMath.add(products.stream().map(ProductEntity::getPrice).toArray(BigDecimal[]::new)); } } ``` 这里展示了如何在一个服务层函数内集成之前编写的 `BigDecimal` 辅助类完成业务需求[^2]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值