一.为什么要用到BigDecimal?
1.首先上一个小例子 以下这段代码,大家猜一下运行结果是什么?
double test1 = 0.01;
double test2 = 0.03;
double testCountValue = test2-test1;
System.out.println("相减的值为----"+testCountValue);
一眼看上去是不是以为是0.02 其实是错的 正确答案是 0.019999999999999997 因为缺失精度了 ,如果这只是个测试例子还好,要是是钱可就。。。。
二.了解什么是BigDecimal?
1.在商业计算中要用java.math.BigDecimal 所创建的是对象, 关于BigDecimal对象我们不能使用传统的+ - * /等算数运算符直接降级对象进行数学运算,而必须调用其相对应的方法,对应的数据库里面存的字段类型可以是decimal。
三.怎么使用BigDecimal?
1.如何定义?
BigDecimal 变量名 = new BigDecimal(数据类型);
数据类型里面可以是 :String ,int double long
2.BigDecimal类型如何转换成其他类型?
toString() 将BigDecimal对象的数值转换成字符串。
doubleValue() 将BigDecimal对象中的值以双精度数返回。
floatValue() 将BigDecimal对象中的值以单精度数返回。
longValue() 将BigDecimal对象中的值以长整数返回。
intValue() 将BigDecimal对象中的值以整数返回。
四.BigDecimal的常用方法:
加法 :add()
减法: substract()
乘法:multiply()
除法:divide()
设置精度: setScale(int newScale, RoundingMode roundingMode);
注:newScale小数点后面保留位数, roundingMode取舍方式
比较:compareTo()
求余数,求b1除以b2的余数:remainder(b2):
最大数,求两个BigDecimal类型数据的最大值:b1.max(b2) :
b1.min(b2) : 最小数,求两个BigDecimal类型数据的最小值。
bi.abs():绝对值,求BigDecimal类型数据的绝对值。
b1.negate():相反数,求BigDecimal类型数据的相反数。
对于乘法和除法,为了得到精确的小数位数,需要设置精度,通过setScale(int newScale, RoundingMode roundingMode)进行设置,newScale为需要保留的小数点位数,roundingMode精确方式,其中,除法如果除不尽,必须设置精度,否则报错。
五.代码演示
DecimalFormat decimalFormat = new DecimalFormat("###0.00");
double testValue1 = 0.111;
double testValue2 = 0.999;
BigDecimal testBigDecimal1 = new BigDecimal(decimalFormat.format(testValue1));
BigDecimal testBigDecimal2 = new BigDecimal(decimalFormat.format(testValue2));
//加法(输出为 1.11),注意testValue1原本的值没变,还是0.111
System.out.println("俩者相加的值为:--"+testBigDecimal1.add(testBigDecimal2).doubleValue());
System.out.println("111--"+testValue1);
//减法(输出为0.89)
System.out.println("俩者相减的值为:---"+testBigDecimal2.subtract(testBigDecimal1));
//乘法,保留两位小数(输出的值为 0.11)
System.out.println("俩者相乘的值为:---"+testBigDecimal1.multiply(testBigDecimal2).setScale(2, BigDecimal.ROUND_HALF_UP).doubleValue());
//除法(输出的值为:0.11)
System.out.println("俩者相除的值为:---"+testBigDecimal1.divide(testBigDecimal2).setScale(2, BigDecimal.ROUND_HALF_UP).doubleValue());
//比较, 在比较BigDecimal的时候,如果返回的结果是小于 返回-1,等于为0 ,大于为1
BigDecimal one = BigDecimal.valueOf(1);
BigDecimal two = BigDecimal.valueOf(2);
BigDecimal three = one.add(two);
System.out.println(one.compareTo(two));//输出-1
System.out.println(two.compareTo(two));//输出 0
System.out.println(three.compareTo(two));//输出 1
其中常用的精确方式有:可以参考这篇:
* ROUND_HALF_UP 四舍五入 2.35变成2.4
* ROUND_CEILING 向正无穷方向舍入
* ROUND_DOWN 向零方向舍入 直接删除多余的小数位,如2.35会变成2.3
* ROUND_FLOOR 向负无穷方向舍入
* ROUND_HALF_DOWN 向(距离)最近的一边舍入
六:最后需要注意的一点是:
new BigDecimal(不能直接是double类型);
举个例子:此时输出的值不是0.1 而是0.1000000000000000055511151231257827021181583404541015625
BigDecimal testDoubleValue = new BigDecimal(0.1);
System.out.println("double 类型输出的值为---"+ testDoubleValue);
这个的解决方案有三种:
BigDecimal testDoubleValue1 = BigDecimal.valueOf(0.1);
System.out.println("第一种解决办法:double 的值为:----"+ testDoubleValue1);
BigDecimal testDoubleValue2 = new BigDecimal(String.valueOf(0.1));
System.out.println("第二种解决办法:double 的值为:----"+ testDoubleValue2);
DecimalFormat df = new DecimalFormat("###0.00");
BigDecimal testDoubleValue3= new BigDecimal(df.format(0.1));
System.out.println("第三种解决办法:保留俩位小数,double 的值为:----"+ testDoubleValue3);