为什么BigDecimal会丢失精度?

为什么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

floatdouble类型主要是为了科学计算和工程计算而设计,他们执行二进制浮点运算。
因为使用二进制计算效率要高,适用于日常运算,而二进制也能给我提供一个相对准确的值。
为什么二进制运算不能提供准确的值,其实不能说二进制不能提供准确的值,而是二进制不能准确的表示一个小数,就像十进制不能准确的表示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));
}

你可能感兴趣的:(java知识)