BigDecimal详解 BigDecimal加减乘除运算 BigDecimal比较大小 BigDecimal保留两位小数

本文详细介绍了Java中BigDecimal类的使用,包括为何使用BigDecimal避免浮点数运算误差, BigDecimal的初始化、加减乘除运算、比较大小、保留小数的方法以及舍入模式。同时,文章提供了丰富的代码示例,展示了如何处理除不尽的情况和如何进行数值比较。此外,还提及了BigDecimal的一些常见方法和常量,如求绝对值、求相反数、乘方等。
摘要由CSDN通过智能技术生成

1、为什么要用BigDecimal ?

工作中我们通过浮点数进行运算时,好像时不时的会出现一些小误差。例如:

public static void main(String[] args) {
    System.out.println(1.9 - 1.2);
    System.out.println(1.9 - 1.5);
    System.out.println(100 - 99.8);
}

在这里插入图片描述
大致搜了一下原因,网上说我们的计算机是二进制的,而浮点数是没有办法通过二进制精准的表示出来。
也就导致在运算的时候,float类型和double类型很容易丢失精度。
所以在开发中,如果我们需要精确计算的结果,可以使用java.math包中提供的BigDecimal类来进行操作。

2、BigDecimal初始化赋值

方法类型描述
public BigDecimal(int val)构造函数int类型的值生成BigDecimal对象
public BigDecimal(long val)构造函数long类型的值生成BigDecimal对象
public BigDecimal(String val)静态方法String类型的值转换为BigDecimal类型
public static BigDecimal valueOf(double val)静态方法double类型的值转换为BigDecimal类型
public static BigDecimal valueOf(long val)静态方法long类型(包含int类型)的值转换为BigDecimal类型
  • 代码示例:
BigDecimal b = new BigDecimal("33");
BigDecimal c = BigDecimal.valueOf(4.7);
  • 注意:不建议使用public BigDecimal(double val)方式初始化值,编码时idea提示禁止使用构造方法BigDecimal(double),描述如下:
使用了new BigDecimal(double)构造函数 less... (Ctrl+F1) 
Inspection info: 
禁止使用构造方法BigDecimal(double)的方式把double值转化为BigDecimal对象 说明:反编译出的字节码文件显示每次循环都会new出一个StringBuilder对象,然后进行append操作,最后通过toString方法返回String对象,造成内存资源浪费。
            
Negative example(不建议使用):
    BigDecimal good1 = new BigDecimal(0.1);

Positive example(建议使用):
    BigDecimal good1 = new BigDecimal("0.1");
    BigDecimal good2 = BigDecimal.valueOf(0.1);

在这里插入图片描述

3、BigDecimal的加减乘除运算

运算法则对应方法
加法public BigDecimal add(BigDecimal value)
减法public BigDecimal subtract(BigDecimal value)
乘法public BigDecimal multiply(BigDecimal value)
除法public BigDecimal divide(BigDecimal value)
  • 代码示例:
public static void main(String[] args) {
    System.out.println("计算加法: " + BigDecimal.valueOf(1.9).add(BigDecimal.valueOf(0.2)));
    System.out.println("计算减法: " + BigDecimal.valueOf(1.9).subtract(BigDecimal.valueOf(1.5)));
    System.out.println("计算乘法: " + BigDecimal.valueOf(1.9).multiply(BigDecimal.valueOf(0.2)));
    System.out.println("计算除法: " + BigDecimal.valueOf(1.9).divide(BigDecimal.valueOf(0.2)));
}

在这里插入图片描述

  • 注意1:BigDecimal的运算结果都是返回了一个新的BigDecimal对象,并不是在原有的对象上进行操作。
public static void main(String[] args) {
    BigDecimal a = BigDecimal.valueOf(5);
    System.out.println("a的地址:" + System.identityHashCode(a));
    a = a.add(BigDecimal.valueOf(3));
    System.out.println("计算后a的地址:" + System.identityHashCode(a));
}

在这里插入图片描述

  • 注意2:使用divide除法函数除不尽,出现无线循环小数的时候,就需要使用另外精确的小数位数以及舍入模式,不然会出现报错。例如:
public static void main(String[] args) {
    BigDecimal a = BigDecimal.valueOf(10), b = BigDecimal.valueOf(3);
    System.out.println(a.divide(b));
}

// 该程序运行会出现以下错误
Exception in thread "main" java.lang.ArithmeticException: Non-terminating decimal expansion; no exact representable decimal result.
    at java.math.BigDecimal.divide(BigDecimal.java:1690)
    at com.fivesix._05_bigdecimal.Demo01.main(Demo01.java:31)

解决方法如下(此处舍入模式使用四舍五入的方式,其他模式在该文章后面有讲解):

public static void main(String[] args) {
    BigDecimal a = BigDecimal.valueOf(10), b = BigDecimal.valueOf(3);
    System.out.println(a.divide(b, 3, BigDecimal.ROUND_HALF_UP));
}
// 该程序运行后输出:
3.33

4、BigDecimal比较大小

public int compareTo(BigDecimal val)

BigDecimal类提供的比较值的方法,注意比较的两个值均不能为空
a.compareTo(b)得到结果 1, 0, -1。

比较结果描述
1a 大于b
0a 等于b
-1a 小于b
  • 代码示例:
public static void main(String[] args) {
    BigDecimal a = BigDecimal.valueOf(1);
    BigDecimal b = BigDecimal.valueOf(2);
    BigDecimal c = BigDecimal.valueOf(1);
    BigDecimal d = BigDecimal.ZERO;
    System.out.println("1和2比较结果:" + a.compareTo(b));
    System.out.println("1和1比较结果:" + a.compareTo(c));
    System.out.println("1和0比较判断:" + (a.compareTo(d) > 0) );
}

在这里插入图片描述

5、BigDecimal保留两位小数及舍入模式

public BigDecimal setScale(int newScale, int roundingMode)

用于格式化小数的方法,第一个值表示保留几位小数,第二个值表示格式化的类型。
8种类型如下:

格式化类型描述
ROUND_DOWN舍弃多余位数,如1.55会格式化为1.5,-1.55会格式化为-1.5
ROUND_UP进位处理,如1.52会格式化为1.6,-1.52会格式化为-1.6
ROUND_HALF_UP四舍五入,如果舍弃部分>= .5,则进位
ROUND_HALF_DOWN五舍六入,如果舍弃部分> .5,则进位
ROUND_CEILING正无穷大方向舍入模式。如果值为正数,则与ROUND_UP模式相同;如果值为负数,则与ROUND_DOWN模式相同
ROUND_FLOOR负无穷大方向舍入模式。如果值为正数,则与ROUND_DOWN模式相同;如果值为负数,则与ROUND_UP模式相同
ROUND_UNNECESSARY确认值的小数位数是否与传入第一个参数(保留小数的位数)相等,如果符合则返回值,如果不符抛出异常
ROUND_HALF_EVEN如果舍弃部门左边的数字为奇数,则与ROUND_HALF_UP模式相同,如果为偶数则与ROUND_HALF_DOWN模式相同
  • 代码示例:
public static void main(String[] args) {
    BigDecimal a = BigDecimal.valueOf(5.445);
    System.out.println("5.445舍弃多余位数:" + a.setScale(2, BigDecimal.ROUND_DOWN));
    System.out.println("5.445进位处理:" + a.setScale(2, BigDecimal.ROUND_UP));
    System.out.println("5.445四舍五入(舍弃部分>= .5,进位):" + a.setScale(2, BigDecimal.ROUND_HALF_UP));
    System.out.println("5.445四舍五入(舍弃部分未> .5,舍弃):" + a.setScale(2, BigDecimal.ROUND_HALF_DOWN));
    System.out.println("5.446四舍五入(舍弃部分> .5,进位):" + BigDecimal.valueOf(5.446).setScale(2, BigDecimal.ROUND_HALF_DOWN));
}

在这里插入图片描述

6、BigDecimal其他方法及常量

代码类型描述
BigDecimal.ZERO常量初始化一个为0的BigDecimal对象
BigDecimal.ONE常量初始化一个为1的BigDecimal对象
BigDecimal.TEN常量初始化一个为10的BigDecimal对象
public BigDecimal abs()方法求绝对值,不管正数还是负数,都得到正数
public BigDecimal negate()方法求相反数,正变负,负变正
public BigDecimal pow(int n)方法求乘方,如BigDecimal.valueOf(2).pow(3)的值为8
public BigDecimal max(BigDecimal val)方法两值比较,返回最大值
public BigDecimal min(BigDecimal val)方法两值比较,返回最小值
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值