浮点数计算钱

阅读更多

一..为什么不用浮点数算钱

1. 浮点运算在计算过程中需要对阶,而在对阶过程中可能损失精度,即尾数部分被舍弃。

public class TestDouble {
	public static void main(String[] args) {
		Double a = 1.0e15;
		for (int i = 0; i < 1000000; i++) {
			a += 0.01d;
		}
		System.out.println(a);
		DecimalFormat decimalFormat = new DecimalFormat("0.00");
		System.out.println(decimalFormat.format(a));
	}
}

 结果如下

1.0E15
1000000000000000.00

 实际结果应该是

public static void main(String[] args) {
		BigDecimal bigDecimal = new BigDecimal("1000000000000000");
		BigDecimal add = new BigDecimal("0.01");
		for (int i = 0; i < 1000000; i++) {
			bigDecimal = bigDecimal.add(add);
		}
		System.out.println(bigDecimal.toPlainString());
	}

 

1000000000010000.00

 少算了10000元。

如果把1.0e15改成1.0e8即1千万,则相差0.01,但如果循环次数变得更多,则相差更大。

 

二.为了解决这种问题,JAVA提供了BigDecimal来解决高精度计算问题。

它有两个优点:

1.精度比浮点数高得多,可以指定任一精度,不会像浮点数一样因为对阶造成精度丢失。

2.符合十进制的运算规则,可指定舍入策略,比如四舍五入等。

需要注意的点:

1.初始化时,最好用字符串。用数字的话,小数数字可能不精确,因为有限的十进制小数,在二进制中可能变为无限小数,见BigDecimal的构造函数解释

2.在除法操作时,要指定精度和舍入策略,如果不设定传入策略,当出现除不尽的情况时会报异常。

   

 BigDecimal divide(BigDecimal divisor, int scale, RoundingMode roundingMode)
          返回一个 BigDecimal,其值为 (this / divisor),其标度为指定标度。
 BigDecimal divide(BigDecimal divisor, MathContext mc)
          返回其值为 (this / divisor)BigDecimal(根据上下文设置进行舍入)。

3.BigDecimal是不可变对象,每次运算的结果都体现在返回的BigDecimal对象上,原对象不变。

  

 

 

你可能感兴趣的:(浮点数,float,double,bigdecimal)