为什么BigDecimal会丢失精度?
BigDecimal b1 = new BigDecimal(0.1);
BigDecimal b2 = new BigDecimal(0.5);
System.out.println("b1="+b1+"\nb2="+b2);
---------------结果----------------------
b1=0.1000000000000000055511151231257827021181583404541015625
b2=0.5
float和double类型主要是为了科学计算和工程计算而设计,他们执行二进制浮点运算。
因为使用二进制计算效率要高,适用于日常运算,而二进制也能给我提供一个相对准确的值。
为什么二进制运算不能提供准确的值,其实不能说二进制不能提供准确的值,而是二进制不能准确的表示一个小数,就像十进制不能准确的表示1/3等。
0.1 * 2 = 0.2 -----0
0.2 * 2 = 0.4 -----0
0.4 * 2 = 0.8 -----0
0.8 * 2 = 1.6 -----1
0.6 * 2 = 1.2 -----1
0.2 * 2 = 0.4 -----0
.
.
.一直循环下去
// 0.0001100110011001100110011001100110011001100110011001101
所以二进制表示不了0.1,但是有的小数却可以表示,就像十进制能表示1/2一样,请看0.5
0.5 * 2 = 1 -----1
//所以0.5用二进制表示就是0.1
这也就解释了,我们遇到的坑,new BigDecimal(0.1);返回一个错误的值,而 new BigDecimal(0.5);返回一个准确的值。
从本质上讲 ,只要他的类型是double或者float,它本身就一个不准确的值(例如0.1),这已经跟BigDecimal无关了。我们使用new BigDecimal(Double d);这个构造时,假如入参是0.1,这其中会将0.1转成Double,而此时0.1的值就已经不准确了。
解决方法:
//使用String构造
BigDecimal b1 = new BigDecimal("0.1");
//使用.valueOf
BigDecimal b1 = BigDecimal.valueOf(0.1);
源码:
public static BigDecimal valueOf(double val) {
return new BigDecimal(Double.toString(val));
}