1.引言

  在java语言中,double和float用于二进制浮点型计算,无法得到精确的结果。而BigDecimal则用于精确的计算。不超过16位有效数字(最好是不超过13位)的科学和工程计算,可以使用double和float,但要求精确计或者超过了16位有效数字(超过13位也建议如此)的商业运算则需要使用BigDecimal进行运行,比如金融行业。

  《Effactive Java》第2版第48条中提到,“float和double类型尤其不适合用于货币计算,因为要让一个float或者double精确地标识0.1(或者10的任何其他负数次方值)是不可能的。” 如果不进行运算,使用String表示金额都比double和float强,当然最好的是BigDecimal。金融行业http通信接口可以使用String来表示金额,如果内部接口调用,建议使用BigDecimal。

2.BigDecimal简介

  BigDecimal 由任意精度的整数非标度值 和32 位的整数标度 (scale) 组成。如果为零或正数,则标度是小数点后的位数。如果为负数,则将该数的非标度值乘以 10 的负scale 次幂。因此,BigDecimal表示的数值是(unscaledValue × 10-scale)

3. BigDecimal的构造方法

  分四类,一类是通过另一个BigDecimal对象构造,一类是使用double/long/int/BigInteger类型的数值构造,一类是通过String字符串构造,最后一类是使用静态方法valueOf()构造,静态方法要求参数是double/long/int/BigInteger等。

  JDK API文档中,对于public BigDecimal(double val) 解释如下:

double 转换为 BigDecimal,后者是 double 的二进制浮点值准确的十进制表示形式。返回的 BigDecimal 的标度是使 (10scale × val) 为整数的最小值。

注:

  1. 此构造方法的结果有一定的不可预知性。有人可能认为在 Java 中写入 new BigDecimal(0.1) 所创建的 BigDecimal 正好等于 0.1(非标度值 1,其标度为 1),但是它实际上等于 0.1000000000000000055511151231257827021181583404541015625。这是因为 0.1 无法准确地表示为 double(或者说对于该情况,不能表示为任何有限长度的二进制小数)。这样,传入 到构造方法的值不会正好等于 0.1(虽然表面上等于该值)。
  2. 另一方面,String 构造方法是完全可预知的:写入 new BigDecimal("0.1") 将创建一个 BigDecimal,它正好 等于预期的 0.1。因此,比较而言,通常建议优先使用 String 构造方法
  3. double 必须用作 BigDecimal 的源时,请注意,此构造方法提供了一个准确转换;它不提供与以下操作相同的结果:先使用 Double.toString(double) 方法,然后使用 BigDecimal(String) 构造方法,将 double 转换为 String。要获取该结果,请使用 static valueOf(double) 方法。

对于public static BigDecimal valueOf(long val)的解释如下:

long 值转换为具有零标度的 BigDecimal。提供的此“静态工厂方法”优先于 (long) 构造方法,因为前者允许重用经常使用的 BigDecimal 值。

故此,对于金融行业中的使用有以下结论:

1) 优先使用String参数的构造方法;

2)如果需要通过double构造BigDecimal,有限使用valueOf(double val)方法;这样得到的BigDecimal和String参数的构造方法标度一致(即标度可控);

3)valueOf()的静态工厂方法优先于long、int等类型参数的构造方法,

4)如果想获取指定标度的初始化数值,比如2位小数的0,可以如下使用:

  1. BigDecimal zero1 = new BigDecimal("0.00");

String的构造方法,会根据字符串的小数位来设定标度,其他的构造方法还需要额外调用setScale()来设定标度。

4. BigDecimal的标度、舍入方式

BigDecimal类的标度使用scale属性表示,就是小数位数,精度使用precision属性表示,就是有效位数(亦即整数位+小数位).

舍入方式有多种,BigDecimal类本身有舍入方式的静态成员字段,但是已经废弃,建议使用RoundingMode枚举值。

RoundingMode枚举值如下:

枚举常量摘要
CEILING
          向正无限大方向舍入的舍入模式。
DOWN

          向零方向舍入的舍入模式。
FLOOR

          向负无限大方向舍入的舍入模式。
HALF_DOWN

          向最接近数字方向舍入的舍入模式,如果与两个相邻数字的距离相等,则向下舍入。
HALF_EVEN

          向最接近数字方向舍入的舍入模式,如果与两个相邻数字的距离相等,则向相邻的偶数舍入。
HALF_UP

          向最接近数字方向舍入的舍入模式,如果与两个相邻数字的距离相等,则向上舍入。
UNNECESSARY

          用于断言请求的操作具有精确结果的舍入模式,因此不需要舍入。
UP

          远离零方向舍入的舍入模式。

其中HALF_UP就是最常用的四舍五入,其他根据英文名字很好理解。

5.BigDecimal的四则运算

如果只是对货币金额值的存取,使用String也可以,但如果金额要参与四则运算,则必须使用BigDecimal进行精确的运算。BigDecimal与double、float的运算相比,BigDecimal运算比double、float运算精确,但速度没有double、float的快。

BigDecimal的四则运算使用以下方法:

  public BigDecimal add(BigDecimal value); //加法

  public BigDecimal subtract(BigDecimal value); //减法

  public BigDecimal multiply(BigDecimal value); //乘法

  public BigDecimal divide(BigDecimal value); //除法

JDK API中对四则运算的标度解释如下:

对于所有算术运算符,运算的执行方式是,首先计算准确的中间结果,然后,使用选择的舍入模式将其舍入为精度设置(如有必要)指定的位数。如果不返回准确结果,则将丢弃准确结果的某些数位。当舍入增加了返回结果的大小时,前导数字“9”的进位传播可能会创建新的数位。例如,将值 999.9 舍入为三位数字,则在数值上等于一千,表示为 100×101。在这种情况下,新的 "1" 是返回结果的前导数位。

除了逻辑的准确结果外,每种算术运算都有一个表示结果的首选标度。下表列出了每个运算的首选标度。

算术运算结果的首选标度

运算 结果的首选标度
max(addend.scale(), augend.scale())
max(minuend.scale(), subtrahend.scale())
multiplier.scale() + multiplicand.scale()
dividend.scale() - divisor.scale()

这些标度是返回准确算术结果的方法使用的标度;准确相除可能必须使用较大的标度除外,因为准确的结果可能有较多的位数。例如,1/32 得到 0.03125

舍入之前,逻辑的准确中间结果的标度是该运算的首选标度。

由上面可以看出来,如果调用上述四个方法的话,计算结果的标度可能不受控制,BigDecimal还为每个运算提供了几个重载方法,可以控制运算结果的标度和舍入方式。

以divide为例进行解释:

JDK API对public BigDecimal divide(BigDecimal divisor) 的解释如下:

返回一个 BigDecimal,其值为 (this / divisor),其首选标度为 (this.scale() - divisor.scale());如果无法表示准确的商值(因为它有无穷的十进制扩展),则抛出 ArithmeticException。 
参数:
divisor - 此 BigDecimal 要相除的值。
返回:
this / divisor
抛出:
ArithmeticException - 如果准确的商值没有无穷的十进制扩展
从以下版本开始:
1.5 
  1. 可以看出,无法精确表示商值时就会抛异常,而除法很大可能无法精确表示商值。所以,必须指定标度和舍入方式。以下几个重载方法可以供调用者指定标度和舍入方式。
    public BigDecimal divide(BigDecimal divisor, int scale, RoundingMode roundingMode)
  1. public BigDecimal divide(BigDecimal divisor, RoundingMode roundingMode)
  1. public BigDecimal divide(BigDecimal divisor, MathContext mc)
    这些方法也可能抛出异常,但异常的情况是:divisor==0 或者 roundingMode==ROUND_UNNECESSARY this.scale() 不足以准确地表示相除的结果。所以只要设置好标度和舍入方式,基本可以保证不会抛异常。
    6. BigDecimal的不可变性
    BigDecimal有一个类似String的特性,就是不可变性。调用setScale或者add等运算方法后,原对象的scale和数值都不会改变,所以需要将方法的返回值进行保存使用。
    比如:a = a.setScale(2); a = a.add(b). 其中abBigDecimal对象。
    7.总结
    1)金融行业的金额一定要使用BigDecimal来表示;
    2BigDecimal最好使用String的构造方法创建,如果要使用double等数值作为参数,也要使用valueOf来创建对象。
    3BigDecimal进行四则运算时,最好指定其标度和舍入方式,否则可能抛异常。
    4BigDecimal是不可变的;四则运算后要使用对象保存结果;
    5)多阅读相关API文档,写测试代码来验证其相关方法的调动,才能更好的掌握。
  2.  
  3. 参考资料:
    Java BigDecimal详解 http://blog.csdn.net/jackiehff/article/details/8582449)
    Effactive Java

使用BigDecimal进行精确运算(http://www.cnblogs.com/chenssy/archive/2012/09/09/2677279.html)

java中BigDecimal在金融行业中的使用的更多相关文章

  1. java中BigDecimal加减乘除基本用法

    Java在java.math包中提供的API类BigDecimal,用来对超过16位有效位的数进行精确的运算.双精度浮点型变量double可以处理16位有效数. 在实际应用中,需要对更大或者更小的数进 ...

  2. Java中BigDecimal的8种舍入模式

    java.math.BigDecimal 不可变的.任意精度的有符号十进制数.BigDecimal 由任意精度的整数非标度值和32位的整数标度(scale)组成. 如果为零或正数,则标度是小数点后的位 ...

  3. Java中BigDecimal的8种舍入模式是怎样的

    Java中BigDecimal的8种舍入模式是怎样的?下面长沙欧柏泰克软件学院和大家一起来学习下吧:  java.math.BigDecimal 不可变的.任意精度的有符号十进制数.BigDecima ...

  4. Java中BigDecimal的舍入模式

    java.math.BigDecimal 不可变的.任意精度的有符号十进制数.BigDecimal 由任意精度的整数非标度值和32位的整数标度(scale)组成. 如果为零或正数,则标度是小数点后的位 ...

  5. Java中BigDecimal类介绍及用法

    Java中提供了大数字(超过16位有效位)的操作类,即 java.math.BinInteger 类和 java.math.BigDecimal 类,用于高精度计算. 其中 BigInteger 类是 ...

  6. [转]Java中BigDecimal的使用

    原文地址:https://blog.csdn.net/cen_s/article/details/76472834 在日常开发中我们经常会碰到小数计算,而小数直接计算的话会出现一些小小的错误,如下 S ...

  7. java中BigDecimal的学习

    干着java的活,但是看的都是一些偏底层的东西(或者我根本就没有看),有点荒废了java的学习. 最近一直在用到一个类是BigDecimal,但都是模棱两可地在那儿用,并没有深入研究这个类的细节,感觉 ...

  8. 商业计算中Java高精度计算BigDecimal类

    <Effective Java> 第48条:如果需要精确的答案,请避免使用float和double. 如果我们编译运行下面这个程序会看到什么?public class Test{    p ...

  9. Java基础扫盲系列(二)—— Java中BigDecimal和浮点类型

    一直以来我几乎未使用过BigDecimal类型,只有在DB中涉及到金额字段时听说要用Decimal类型,但是今天再项目代码中看到使用BigDecimal表示贷款金额. 本篇文章不是介绍BigDecim ...

随机推荐

  1. 从无到有之webpack+vuerouter的简单例子以及各个属性解释

    之前一直没玩过webpack和vue,近两周才看这玩意,本文纯属自己的实验+之前angular作战经验的理解一些入门文章 首先webpack关于vue以及各个包 module.exports = { ...

  2. 超级迷宫需求分析与建议-NABCD模型

    超级迷宫需求分析与建议-NABCD模型 制作者-姜中希 1N-Need 需求  首先这是一个手机游戏风靡的时代,随着智能手机不断的更新问世,4G网络的不断扩大普及,越来越多的手机游戏受到广大玩家的追捧 ...

  3. 【树上DFS】Tree and Polynomials

    http://codeforces.com/gym/101372 D push1[i][k]:所有操作1总共要让节点i下推多少系数k push2[i][k]:所有操作2总共要让节点i上推多少系数k s ...

  4. [转]Android试验:如果View的ID相同会出现什么效果?

    1.实验:通过布局编辑器强行指定两个button id相同,然后在代码中通过findViewById()获得句柄后修改其文本.  实验结果:只有一个button的文本变化了,另一个不受影响.  2.实 ...

  5. alpha6/10

    队名:Boy Next Door 燃尽图 晗(组长) 今日完成 学习了css的一些基本操作. 明日工作 抽空把javascript的基本操作学习一下 还剩下哪些任务 微信API还有京东钱包的API. ...

  6. Spring中jdbc Template使用

    http://1358440610-qq-com.iteye.com/blog/1826816

  7. Google Professional Data Engineer(PDE)考试

    在国内参加PDE考试的人比较少,导致资料也很少.我在19年1月30号去上海参加PDE考试,参加前也是完全没底,因为时间短资料少,但幸运的是顺利通过了.回过头来看,其中有些技巧和重点,在此做一些总结,希 ...

  8. 开发环境解决 kafka Failed to send messages after 3 tries

    新建了一个kafka集群,在window下写了一个简单的producer做测试,结果遇到了消息发送失败的问题,代码如下: Properties props = new Properties(); pr ...

  9. SQL之case when then用法详解

    case具有两种格式.简单case函数和case搜索函数. <span style="font-size:14px;">--简单case函数 case sex when ...

  10. [mysqld_safe]centos7 mysql 安装与配置

    查资料发现是CentOS 7 版本将MySQL数据库软件从默认的程序列表中移除,用mariadb代替了. 有两种解决办法: 安装mariadb [root@a ~]#  yum install mar ...