float 和 double 在内存中以科学计数法的形式存储,在转换过程中容易发生精度丢失。尤其在进行算术运算时更容易发生这种情况。
可以使用BIgDecimal类来实现对浮点数的精确表示和计算。
需要注意的是创建BigDecimal对象时,不要直接使用double浮点数作为构造器参数来调用BigDecimal构造器,否则同样会发生精度丢失的问题:
import java.math.BigDecimal;
public class BigDecimalTest {
public static void main(String[] args) {
BigDecimal f1 = new BigDecimal("0.05");
BigDecimal f2 = BigDecimal.valueOf(0.01);
BigDecimal f3 = new BigDecimal(0.05);
System.out.println("使用String作为BigDecimal构造器参数:");
System.out.println(" 0.05 + 0.01 = " + f1.add(f2));
System.out.println(" 0.05 - 0.01 = " + f1.subtract(f2));
System.out.println(" 0.05 * 0.01 = " + f1.multiply(f2));
System.out.println(" 0.05 / 0.01 = " + f1.divide(f2));
System.out.println("使用double作为BigDecimal构造器参数:");
System.out.println(" 0.05 + 0.01 = " + f3.add(f2));
System.out.println(" 0.05 - 0.01 = " + f3.subtract(f2));
System.out.println(" 0.05 * 0.01 = " + f3.multiply(f2));
System.out.println(" 0.05 / 0.01 = " + f3.divide(f2));
}
}
输出结果:
使用String作为BigDecimal构造器参数:
0.05 + 0.01 = 0.06
0.05 - 0.01 = 0.04
0.05 * 0.01 = 0.0005
0.05 / 0.01 = 5
使用double作为BigDecimal构造器参数:
0.05 + 0.01 = 0.06000000000000000277555756156289135105907917022705078125
0.05 - 0.01 = 0.04000000000000000277555756156289135105907917022705078125
0.05 * 0.01 = 0.0005000000000000000277555756156289135105907917022705078125
0.05 / 0.01 = 5.000000000000000277555756156289135105907917022705078125
因此应当优先使用基于String的构造器,如果必须使用double浮点数作为BigDecimal构造器的参数时,不要直接将该double浮点数作为构造器参数创建BigDecimal对象,而是应该通过BigDecimal.valueOf(double value)静态方法来创建BigDecimal对象:
BigDecimal f2 = BigDecimal.valueOf(0.01);