double类型精度丢失问题:
0.1*0.1使用计算器计算是0.01,代码里却是0.010000000000000002
public class HelloWorld {
public static void main(String []args) {
double number1 = 0.1;
double number2 = 0.1;
double result = number1 * number2 ;
System.out.println("使用double运算结果: "+result);
}
}
为什么会这样呢?这就是精度丢失问题造成的。
为什么会出现精度丢失?
因为计算机只能识别0和1,即二进制,无论哪种编程语言,都需要翻译成二进制才能被计算机识别。
很多人还知道这样一句话:这种舍入误差的主要原因是浮点数值采用二进制系统表示, 而在二进制系统中无法精确地表示分数 1/10。这就好像十进制无法精确地表示分数 1/3—样。
针对十进制,1除以3是除不尽的。很好理解,因为我们一直接触的就是十进制,等于0.333333… 很好理解
但是:二进制系统中无法精确地表示分数 1/10。为啥呢。就有点不理解了
《Java核心技术卷》书上也是这么写的。
十进制 转二进制(每次将小数部分乘2,取出整数部分,如果小数部分为0,就可以停止这个过程):十进制0.1
0.1*2=0.2
0.2*2=0.4
0.4*2=0.8
0.8*2=1.6
0.6*2=1.2
0.2*2=0.4
0.4*2=0.8
//... 应该已经发现,上面的过程已经开始循环,小数部分永远不能为0
怎么解决精度丢失问题?
在商城里面计算订单金额的时候,我们就不得不解决这个问题了,这时候就用到了BigDecimal
BigDecimal类位于java.math包下,用于对超过16位有效位的数进行精确的运算。
一般来说,double类型的变量可以处理16位有效数,
但实际应用中,如果超过16位,就需要BigDecimal类来操作
new BigDecimal(double val)
new BigDecimal(String val)
BigDecimal.valueOf(double val)