BigDecimal详解

本文主要描述BigDecimal的用法,BigDecimal类型可以进行精确计算应用场景非常广泛,在商业计算中可以说是必用。

1.介绍

Java在java.math包中提供的API类BigDecimal,用来对超过16位有效位的数进行精确的运算。双精度浮点型变量double可以处理16位有效数。在实际应用中,需要对更大或者更小的数进行运算和处理。float和double只能用来做科学计算或者是工程计算,在商业计算中要用java.math.BigDecimal。

 
2.BigDecimal用法
2.1构造方法

BigDecimal是一个对象,所以有其构造方法。形式如下:

BigDecimal(int)          创建一个具有参数所指定整数值的对象。 
BigDecimal(double)       创建一个具有参数所指定双精度值的对象。 
BigDecimal(long)         创建一个具有参数所指定长整数值的对象。 
BigDecimal(String)       创建一个具有参数所指定以字符串表示的数值的对象。

通常建议优先使用String构造方法。尤其是使用BigDecimal(double)容易出错,示例如下:

BigDecimal aDouble =new BigDecimal(1.22);
System.out.println("construct with a double value: " + aDouble);
BigDecimal aString = new BigDecimal("1.22");
System.out.println("construct with a String value: " + aString);
输出结果如下:
construct with a doublevalue:1.2199999999999999733546474089962430298328399658203125
construct with a String value: 1.22
2.3运算方法

BigDecimal所创建的是对象,我们不能使用传统的+、-、*、/等算术运算符直接对其对象进行数学运算,而必须调用其相对应的方法。其常用的方法如下:

add(BigDecimal)             BigDecimal对象中的值相加,然后返回这个对象。 
subtract(BigDecimal)        BigDecimal对象中的值相减,然后返回这个对象。 
multiply(BigDecimal)        BigDecimal对象中的值相乘,然后返回这个对象。 
divide(BigDecimal)          BigDecimal对象中的值相除,然后返回这个对象。 
toString()                  将BigDecimal对象的数值转换成字符串。 
doubleValue()               将BigDecimal对象中的值以双精度数返回。 
floatValue()                将BigDecimal对象中的值以单精度数返回。 
longValue()                 将BigDecimal对象中的值以长整数返回。 
intValue()                  将BigDecimal对象中的值以整数返回。

其中除法(divide)运算需要注意,BigDecimal除法可能出现不能整除的情况,比如 1/3,这时会报错java.lang.ArithmeticException: Non-terminating decimal expansion; no exact representable decimal result.这时就需要提前做处理。
divide方法可以传三个参数:
public BigDecimal divide(BigDecimal divisor, int scale, int roundingMode)
第一个参数表示除数
第二个参数表示小数点后保留位数
第三个参数表示舍入模式,只有在作除法运算或四舍五入时才用到舍入模式
示例:

String money = new BigDecimal("1").divide(new BigDecimal("3"),2,BigDecimal.ROUND_HALF_UP).toString();
String price = new BigDecimal("20").divide(new BigDecimal("100")).setScale(2, BigDecimal.ROUND_HALF_UP).toString();
2.4八种舍入模式

(1)ROUND_UP
  向远离零的方向舍入。舍弃非零部分,并将非零舍弃部分相邻的一位数字加一。

(2)ROUND_DOWN
  向接近零的方向舍入。舍弃非零部分,同时不会非零舍弃部分相邻的一位数字加一,采取截取行为。

(3)ROUND_CEILING
  向正无穷的方向舍入。如果为正数,舍入结果同ROUND_UP一致;如果为负数,舍入结果同ROUND_DOWN一致。注意:此模式不会减少数值大小。

(4)ROUND_FLOOR
  向负无穷的方向舍入。如果为正数,舍入结果同ROUND_DOWN一致;如果为负数,舍入结果同ROUND_UP一致。注意:此模式不会增加数值大小。

(5)ROUND_HALF_UP
  向“最接近”的数字舍入,如果与两个相邻数字的距离相等,则为向上舍入的舍入模式。如果舍弃部分>= 0.5,则舍入行为与ROUND_UP相同;否则舍入行为与ROUND_DOWN相同。这种模式也就是我们常说的我们的“四舍五入”。

(6)ROUND_HALF_DOWN
  向“最接近”的数字舍入,如果与两个相邻数字的距离相等,则为向下舍入的舍入模式。如果舍弃部分> 0.5,则舍入行为与ROUND_UP相同;否则舍入行为与ROUND_DOWN相同。这种模式也就是我们常说的我们的“五舍六入”。

(7)ROUND_HALF_EVEN
  向“最接近”的数字舍入,如果与两个相邻数字的距离相等,则相邻的偶数舍入。如果舍弃部分左边的数字奇数,则舍入行为与 ROUND_HALF_UP 相同;如果为偶数,则舍入行为与 ROUND_HALF_DOWN 相同。注意:在重复进行一系列计算时,此舍入模式可以将累加错误减到最小。此舍入模式也称为“银行家舍入法”,主要在美国使用。四舍六入,五分两种情况,如果前一位为奇数,则入位,否则舍去。

(8)ROUND_UNNECESSARY
  断言请求的操作具有精确的结果,因此不需要舍入。如果对获得精确结果的操作指定此舍入模式,则抛出ArithmeticException。

下面,举个例子说明一下不同舍入模式下的数值计算结果,保留一位小数:
八种舍入模式示例

2.5格式化及例子

由于NumberFormat类的format()方法可以使用BigDecimal对象作为其参数,可以利用BigDecimal对超出16位有效数字的货币值,百分值,以及一般数值进行格式化控制。代码示例如下:

public static void main(String[] args) {
    NumberFormat currency = NumberFormat.getCurrencyInstance(); //建立货币格式化引用 
    NumberFormat percent = NumberFormat.getPercentInstance();  //建立百分比格式化引用 
    percent.setMaximumFractionDigits(3); //百分比小数点最多3位 
    
    BigDecimal loanAmount = new BigDecimal("15000.48"); //贷款金额
    BigDecimal interestRate = new BigDecimal("0.008"); //利率   
    BigDecimal interest = loanAmount.multiply(interestRate); //相乘

    System.out.println("贷款金额:\t" + currency.format(loanAmount)); 
    System.out.println("利率:\t" + percent.format(interestRate)); 
    System.out.println("利息:\t" + currency.format(interest)); 
}
运行结果如下:
贷款金额:15,000.48
利率:         0.8%
利息:120.00
2.6BigDecimal比较

BigDecimal是通过使用compareTo(BigDecimal)来比较的,具体比较情况如下:

Boolean flag =new BigDecimal("100").compareTo(BigDecimal.ZERO)==0;     //true代表等于0
Boolean flag =new BigDecimal("100").compareTo(BigDecimal.ZERO)==1;     //true代表大于0
Boolean flag =new BigDecimal("100").compareTo(BigDecimal.ZERO)==-1;    //true代表小于0

常用new BigDecimal(“100”).compareTo(BigDecimal.ZERO)!=1(金额中判断小于等于零的情况)

2.7toPlainString和toString的区别
System.out.println(new BigDecimal("0.0000001").toString;              //结果为:1E-7
System.out.println(new BigDecimal("0.0000001").toPlainString());       //结果为:0.0000001

在项目中使用toString时,小数点后边位数少的时候可能没有问题,但小数点后位数多的时候可能就会出现问题,所以使用toString有可能表示的结果为科学计数法,但使用toPlainString展示结果一定正确(不会出现科学计数法)。

  • 3
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值