使用BigDecimal表示金钱,抛弃 double

  • :简介

  • 二:BigDecimal

  • 三:工具类

  • 四 :BigDecimal除法四舍五入的问题

  • 五:测试

  • 六 :注意的问题


一:简介

Java中的简单浮点数类型float和double不能够进行运算,或者运算会丢失精度,不光是Java,在其它很多编程语言中也有这样的问题。在大多数情况下,计算的结果是准确的,float和double只能用来做科学计算或者是工程计算,在商业计算中我们要用 java.math.BigDecimal。BigDecimal类的主要功能是进行小数的大数计算,而且最重要的是可以精确到指定的四舍五入位数

// 浮点型计算不准确示例
@Test
public void testDouble(){
 // 0.060000000000000005
 System.out.println(0.05+0.01);
 // 0.5800000000000001
 System.out.println(1.0-0.42);
 // 401.49999999999994
 System.out.println(4.015*100);
 // 1.2329999999999999
 System.out.println(123.3/100);
 // 4.01 四舍五入保留两位
 System.out.println(Math.round(4.015*100)/100.0);
}

二:BigDecimal

BigDecimal有多种构造函数,常用的有2种,其中有一种不建议使用,就是double构造方式,建议使用String构造方式

// 强制使用String的构造函数,double也有可能计算不太准确
// 原则是使用BigDecimal并且一定要用String来够造。
public BigDecimal(String val);
public BigDecimal(double val);

 

BigDecimal也定义了几个常用的值,0、1、10,静态的,可以通过类名直接引用BigDecimal.ZERO
public static final BigDecimal ZERO = zeroThroughTen[0];
 /**
 * The value 1, with a scale of 0.
 *
 * @since 1.5
 */
 public static final BigDecimal ONE = zeroThroughTen[1];
 /**
 * The value 10, with a scale of 0.
 *
 * @since 1.5
 */
 public static final BigDecimal TEN = zeroThroughTen[10];

三:工具类

由于构造方法要用String对应的构造方法,如果我们要做一个加法运算,需要先将两个浮点数转为String,然后够造成BigDecimal,在其中一个上调用add方法,传入另一个作为参数,然后把运算的结果(BigDecimal)再转换为浮点数。你能够忍受这么烦琐的过程吗?下面我们提供一个工具类来简化操作。

import java.math.BigDecimal;
/**
 * 由于Java的简单类型不能够精确的对浮点数进行运算,这个工具类提供精
 * 确的浮点数运算,包括加减乘除和四舍五入。
 */
public class ArithUtil {
 //默认除法运算精度
 private static final int DEF_DIV_SCALE = 10;
 /**
 * 提供精确的加法运算。
 * @param v1 被加数
 * @param v2 加数
 * @return 两个参数的和
 */
 public static double add(double v1,double v2){
 BigDecimal b1 = new BigDecimal(Double.toString(v1));
 BigDecimal b2 = new BigDecimal(Double.toString(v2));
 return b1.add(b2).doubleValue();
 }
 /**
 * 提供精确的减法运算。
 * @param v1 被减数
 * @param v2 减数
 * @return 两个参数的差
 */
 public static double sub(double v1,double v2){
 BigDecimal b1 = new BigDecimal(Double.toString(v1));
 BigDecimal b2 = new BigDecimal(Double.toString(v2));
 return b1.subtract(b2).doubleValue();
 }
 /**
 * 提供精确的乘法运算。
 * @param v1 被乘数
 * @param v2 乘数
 * @return 两个参数的积
 */
 public static double mul(double v1,double v2){
 BigDecimal b1 = new BigDecimal(Double.toString(v1));
 BigDecimal b2 = new BigDecimal(Double.toString(v2));
 return b1.multiply(b2).doubleValue();
 }
 /**
 * 提供(相对)精确的除法运算,当发生除不尽的情况时,精确到
 * 小数点以后10位,以后的数字四舍五入。
 * @param v1 被除数
 * @param v2 除数
 * @return 两个参数的商
 */
 public static double div(double v1,double v2){
 return div(v1,v2,DEF_DIV_SCALE);
 }
 /**
 * 提供(相对)精确的除法运算。当发生除不尽的情况时,由scale参数指
 * 定精度,以后的数字四舍五入。
 * @param v1 被除数
 * @param v2 除数
 * @param scale 表示表示需要精确到小数点以后几位。
 * @return 两个参数的商
 */
 public static double div(double v1,double v2,int scale){
 if(scale<0){
 throw new IllegalArgumentException("The scale must be a positive integer or zero");
 }
 BigDecimal b1 = new BigDecimal(Double.toString(v1));
 BigDecimal b2 = new BigDecimal(Double.toString(v2));
 return b1.divide(b2,scale,BigDecimal.ROUND_HALF_UP).doubleValue();
 }
 /**
 * 提供精确的小数位四舍五入处理。
 * @param v 需要四舍五入的数字
 * @param scale 小数点后保留几位
 * @return 四舍五入后的结果
 */
 public static double round(double v,int scale){
 if(scale<0){
 throw new IllegalArgumentException("The scale must be a positive integer or zero");
 }
 BigDecimal b = new BigDecimal(Double.toString(v));
 BigDecimal one = new BigDecimal("1");
 return b.divide(one,scale, BigDecimal.ROUND_HALF_UP).doubleValue();
 }
}

四:BigDecimal除法四舍五入

参考:https://www.cnblogs.com/shenwen/p/9481856.html

BigDecimal类的主要功能是进行小数的大数计算,而且最重要的是可以精确到指定的四舍五入位数

如果要进行四舍五入的操作,则必须依靠以下的方法
public BigDecimal divide(BigDecimal divisor,int scale,int roundingMode)
scale:表示四舍五入的位数

BigDecimal和RoundingMode是一个绝配,想要采用什么舍入模式使用RoundingMode设置即可。目前Java支持以下七种舍入方式:

ROUND_UP: 远离零方向舍入。

向远离0的方向舍入,也就是说,向绝对值最大的方向舍入,只要舍弃位非0即进位。

ROUND_DOWN:趋向零方向舍入。

向0方向靠拢,也就是说,向绝对值最小的方向输入,注意:所有的位都舍弃,不存在进位情况。

ROUND_CEILING:向正无穷方向舍入。

向正最大方向靠拢,如果是正数,舍入行为类似于ROUND_UP;如果为负数,则舍入行为类似于ROUND_DOWN。注意:Math.round方法使用的即为此模式。

ROUND_FLOOR:向负无穷方向舍入。

向负无穷方向靠拢,如果是正数,则舍入行为类似于 ROUND_DOWN;如果是负数,则舍入行为类似于 ROUND_UP。

HALF_UP: 最近数字舍入(5进)。

这就是我们最最经典的四舍五入模式。

HALF_DOWN:最近数字舍入(5舍)。

 

在四舍五入中,5是进位的,而在HALF_DOWN中却是舍弃不进位。

HALF_EVEN :银行家算法。

在普通的项目中舍入模式不会有太多影响,可以直接使用Math.round方法,但在大量与货币数字交互的项目中,一定要选择好近似的计算模式,尽量减少因算法不同而造成的损失。(银行家算法就是,假如保留两位,第三位大于5,则进1;小于5,舍去;第三位为奇数,进1,偶数,舍去。)

BigDecimal ba = new BigDecimal(123.015);
       System.out.println("计算123.015/1的结果");
       System.out.println("ROUND_UP 远离零方向舍入 =" + ba.divide(BigDecimal.ONE, 2,BigDecimal.ROUND_UP));
       System.out.println("ROUND_DOWN趋向零方向舍入 =" + ba.divide(BigDecimal.ONE, 2, BigDecimal.ROUND_DOWN));
       System.out.println("ROUND_CEILING向正无穷方向舍入 =" + ba.divide(BigDecimal.ONE, 2, BigDecimal.ROUND_CEILING));
       System.out.println("ROUND_FLOOR向负无穷方向舍入 =" + ba.divide(BigDecimal.ONE, 2, BigDecimal.ROUND_FLOOR));
       System.out.println("ROUND_HALF_UP最近数字舍入(5进) =" + ba.divide(BigDecimal.ONE, 2,BigDecimal.ROUND_HALF_UP));
       System.out.println("ROUND_HALF_DOWN最近数字舍入(5舍) =" + ba.divide(BigDecimal.ONE, 2, BigDecimal.ROUND_HALF_DOWN));
       System.out.println("ROUND_HALF_EVEN银行家算法 =" + ba.divide(BigDecimal.ONE, 2, BigDecimal.ROUND_HALF_EVEN));
       //System.out.println("ROUND_UNNECESSARY =" + ba.divide(BigDecimal.ONE, 2, BigDecimal.ROUND_UNNECESSARY));

结果

五:测试

@Test
public void testBigDecimal(){
 BigDecimal price = BigDecimal.ZERO;
 // 通常建议优先使用(String)构造函数
 BigDecimal amount = new BigDecimal("6.66");
 // 0.06
 System.out.println(ArithUtil.add(0.05, 0.01));
 // 0.58
 System.out.println(ArithUtil.sub(1.0, 0.42));
 // 401.5
 System.out.println(ArithUtil.mul(4.015, 100));
 // 1.233
 System.out.println(ArithUtil.div(123.3, 100));
 // 4.02
 System.out.println(ArithUtil.round(4.015, 2));
 // BigDecimal 比较大小使用compareTo方法
 // public int compareTo(BigDecimal val);
}

六:注意的问题

  1. /** 
  2. * (1)BigInteger和BigDecimal都是不可变(immutable)的,在进行每一步运算时,都会产生一个新的对象,由于创建对象会引起开销, 
  3. * 它们不适合于大量的数学计算,应尽量用long,float,double等基本类型做科学计算或者工程计算。 
  4. * 设计BigInteger和BigDecimal的目的是用来精确地表示大整数和小数,使用于在商业计算中使用。 
  5. * (2)BigDecimal有4个够造方法,其中的两个用BigInteger构造,另一个是用double构造,还有一个使用String构造。 
  6. * 应该避免使用double构造BigDecimal,因为:有些数字用double根本无法精确表示,传给BigDecimal构造方法时就已经不精确了。 
  7. * 比如,new BigDecimal(0.1)得到的值是0.1000000000000000055511151231257827021181583404541015625。 
  8. * 使用new BigDecimal("0.1")得到的值是0.1。因此,如果需要精确计算,用String构造BigDecimal,避免用double构造,尽管它看起来更简单! 
  9. * (3)equals()方法认为0.1和0.1是相等的,返回true,而认为0.10和0.1是不等的,结果返回false。 
  10. * 方法compareTo()则认为0.1与0.1相等,0.10与0.1也相等。所以在从数值上比较两个BigDecimal值时,应该使用compareTo()而不是 equals()。 
  11. * (4)另外还有一些情形,任意精度的小数运算仍不能表示精确结果。例如,1除以9会产生无限循环的小数 .111111...。 
  12. * 出于这个原因,在进行除法运算时,BigDecimal可以让您显式地控制舍入。 
  13. */  
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值