为什么要使用BigDecimal
当我们在Java中用浮点数来加减乘除这些运算之类的,就会造成精确度的丢失
比如用两个double类型来存储-0.02133179和-0.02069092让他们进行相加运算会得到
-0.042022710000000005
但是按照我们的常识来看这个结果一定是错误的,当然和正确值之间的差距那么小也没啥,如果我们用数据的时候乘了几十几百亿那么差距就会很大了
为什么会有误差呢,因为计算机底层存储数据都是用二进制存储的当存储整数的时候不会造成精度损失,但是当存储小数的时候他会一直乘二直到最后结果为0,当我们用double来存储上面两个数的时候就会截取一部分因为double存储的小数位数有限,如果直接输出,展示之类的操作的时候是没有太大问题但是一旦进行运算就会让误差变大.
所以我们对于精确度要求高的操作的时候就必须要使用BigDecimal
BigDecimal
是immutable的,不可变的,只允许读,不允许写
实例化BigDecimal对象
BigDecimal是个类,当我们使用的时候就必须先实例化BigDecimal
BigDeciaml a=new BigDeciaml("0.2");//传入字符串做参数实例化
BigDecimal b=new BigDecimal(0.2);//传入double类型做参数
BigDecimal c=new BigDecimal(2L);//传入long/int类型做参数
BigDecimal d=BigDecimal.valueOf(0.2);
BigDecimal d=BigDecimal.valueOf(2);
一般我们都是用传入字符串来实例化对象
BigDecimal的运算
BigDecimal a = new BigDecimal("1.0");
BigDecimal b = new BigDecimal("0.9");
System.out.println(a.add(b));// 1.9
System.out.println(a.subtract(b));// 0.1
System.out.println(a.multiply(b));// 0.90
System.out.println(a.divide(b));// 无法除尽,抛出 ArithmeticException 异常
System.out.println(a.divide(b, 2, RoundingMode.HALF_UP));// 1.11
我们使用除法运算的时候建议使用三个参数的
对于第三个参数
ROUND_CEILING
向正无限大方向舍入的舍入模式。
ROUND_DOWN
向零方向舍入的舍入模式。
ROUND_FLOOR
向负无限大方向舍入的舍入模式。
ROUND_HALF_DOWN
向最接近数字方向舍入的舍入模式,如果与两个相邻数字的距离相等,则向下舍入。
ROUND_HALF_EVEN
向最接近数字方向舍入的舍入模式,如果与两个相邻数字的距离相等,则向相邻的偶数舍入。
ROUND_HALF_UP
向最接近数字方向舍入的舍入模式,如果与两个相邻数字的距离相等,则向上舍入。
ROUND_UNNECESSARY
用于断言请求的操作具有精确结果的舍入模式,因此不需要舍入。(默认模式)
ROUND_UP
远离零方向舍入的舍入模式。
比较大小
推荐使用compareTo
使用==或者.equals()会出现意想不到的问题
BigDecimal a = new BigDecimal("1.0");
BigDecimal b = new BigDecimal("0.9");
System.out.println(a.compareTo(b));// 1
开根号
public static BigDecimal sqrt(BigDecimal value) {
BigDecimal x = new BigDecimal(Math.sqrt(value.doubleValue()));
return x.add(new BigDecimal(value.subtract(x.multiply(x)).doubleValue() / (x.doubleValue() * 2.0)));
}