BigDecimal 详解

一、常用方法

  1. 初始化数据方法
    • new BigDecimal() 传参支持 integer,long,double,float,BigInteger
    • BigDecimal.ZERO 初始化一个为0的BigDecimal对象
    • BigDecimal.ONE 初始化一个为1的BigDecimal对象
    • BigDecimal.TEN 初始化一个为10的BigDecimal对象

【注意】new BigDecimal()方法初始化一个浮点型数据时,需要显示设置Scale(精度)否则运算时可能会有精度问题(这个和BigDecimal 默认精度有关),如果转成String再初始化,将不受影响

  1. 加法 add(BigDecimal)
  BigDecimal b1 = new BigDecimal("1");
  BigDecimal b2 = new BigDecimal("0.11");
  System.out.println(b1.add(b2));
/*
* 可以试下把传入的字符串改为基本数据类型再进行运算
*/
复制代码
  1. 减法 subtract(BigDecimal)
BigDecimal b1 = new BigDecimal("1");
BigDecimal b2 = new BigDecimal("0.11");
System.out.println(b1.subtract(b2));
复制代码
  1. 乘方 muliply(BigDecimal)
BigDecimal b1 = new BigDecimal("1.11");
BigDecimal b2 = new BigDecimal("0.11");
System.out.println(b1.multiply(b2));
复制代码
  1. 除法 divide(BigDecimal)
  • 做除法运算时需要制定精确度和舍入方式
BigDecimal b1 = new BigDecimal("1.11");
BigDecimal b2 = new BigDecimal("0.11");
//四舍五入保留两位小数 后文将介绍舍入方式
System.out.println(b1.divide(b2,2,BigDecimal.ROUND_HALF_UP));
复制代码
  1. 绝对值 abs()
BigDecimal b1 = new BigDecimal("-1.11");
System.out.println(b1.abs());
复制代码
  1. 相反数 negate()
BigDecimal b1 = new BigDecimal("1.11");
System.out.println(b1.negate());
复制代码
  1. 乘法 pow(int)
BigDecimal b1 = new BigDecimal("1.11");
//二次方
System.out.println(b1.pow(2));
//三次方
System.out.println(b1.pow(3));
复制代码
  1. 精度 scale()和percision()
BigDecimal b1 = new BigDecimal("11.111");
//位数
System.out.println(b1.precision());//5
//小数点后有多少位
System.out.println(b1.scale());//3
复制代码
  1. 比较大小 compareTo(BigDecimal)
BigDecimal b1 = new BigDecimal("1.11");
BigDecimal b2 = new BigDecimal("0.11");
//大于0 则b1>b2;等于0 则b1==b2;小于0 则b1<b2
System.out.println(b1.compareTo(b2));
复制代码
  1. 比较大小并返回大的 max(BigDecimal)
BigDecimal b1 = new BigDecimal("1.11");
BigDecimal b2 = new BigDecimal("0.11");
System.out.println(b1.max(b2));
复制代码
  1. 比较大小并返回小的 min(BigDecimal)
BigDecimal b1 = new BigDecimal("1.11");
BigDecimal b2 = new BigDecimal("0.11");
System.out.println(b1.min(b2));
复制代码

二、舍入方式

  • BigDecimal.ROUND_HALF_UP 四舍五入模式
BigDecimal b1 = new BigDecimal("1.1449");
System.out.println(b1.setScale(2, BigDecimal.ROUND_HALF_UP));//1.14
System.out.println(b1.setScale(3, BigDecimal.ROUND_HALF_UP));//1.145
复制代码
  • BigDecimal.ROUND_HALF_DOWN 五舍六入模式

如果舍去部分大于 0.5 则为进一,如果是小于 0.5 则会舍去

BigDecimal b1 = new BigDecimal("1.1456");
System.out.println(b1.setScale(1, BigDecimal.ROUND_HALF_DOWN));//1.1
System.out.println(b1.setScale(2, BigDecimal.ROUND_HALF_DOWN));//1.15
//上例中舍去部分.56 大于.5 所以会进一
复制代码
BigDecimal b1 = new BigDecimal("1.145");
System.out.println(b1.setScale(1, BigDecimal.ROUND_HALF_DOWN));//1.1
System.out.println(b1.setScale(2, BigDecimal.ROUND_HALF_DOWN));//1.14
//上例舍去部分 .5 不大于.5 所以会舍去
复制代码
  • BigDecimal.ROUND_HALF_EVEN 银行家舍入法 如果舍弃部分左边的数字为奇数,则舍入行为与 ROUND_HALF_UP 相同; 如果为偶数,则舍入行为与 ROUND_HALF_DOWN 相同。
BigDecimal b1 = new BigDecimal("1.25");
System.out.println(b1.setScale(1, BigDecimal.ROUND_HALF_EVEN));//1.2
b1 = new BigDecimal("1.26");
System.out.println(b1.setScale(1, BigDecimal.ROUND_HALF_EVEN));//1.3
b1 = new BigDecimal("1.35");
System.out.println(b1.setScale(1, BigDecimal.ROUND_HALF_EVEN));//1.4
b1 = new BigDecimal("1.36");
System.out.println(b1.setScale(1, BigDecimal.ROUND_HALF_EVEN));//1.4
复制代码
  • BigDecimal.ROUND_UP 始终在保留的最后一位加1
BigDecimal b1 = new BigDecimal("1.1203");
System.out.println(b1.setScale(2, BigDecimal.ROUND_UP));//1.13
System.out.println(b1.setScale(3, BigDecimal.ROUND_UP));//1.121
复制代码

永远不会减少计算值的大小

  • BigDecimal.ROUND_DOWN 从舍弃位置直接截断
BigDecimal b1 = new BigDecimal("1.1209");
System.out.println(b1.setScale(2, BigDecimal.ROUND_DOWN));//1.12
System.out.println(b1.setScale(3, BigDecimal.ROUND_DOWN));//1.120
复制代码

永远不会增加计算值的大小

  • BigDecimal.ROUND_CEILING 接近正无穷大的舍入模式。 如果 BigDecimal 为正,则舍入行为与 ROUND_UP 相同; 如果为负,则舍入行为与 ROUND_DOWN 相同。
BigDecimal b1 = new BigDecimal("1.1209");
System.out.println(b1.setScale(2, BigDecimal.ROUND_CEILING));//1.13
System.out.println(b1.setScale(3, BigDecimal.ROUND_CEILING));//1.121

BigDecimal b2 = b1.negate();//b1的相反数
System.out.println(b2.setScale(2, BigDecimal.ROUND_CEILING));//-1.12
System.out.println(b2.setScale(3, BigDecimal.ROUND_CEILING));//-1.120
复制代码

此舍入模式始终不会减少计算值。

  • BigDecimal.ROUND_FLOOR 和ROUND_CEILING相反 接近负无穷大的舍入模式。 如果 BigDecimal 为正,则舍入行为与 ROUND_DOWN 相同; 如果为负,则舍入行为与 ROUND_UP 相同。
BigDecimal b1 = new BigDecimal("1.1209");
System.out.println(b1.setScale(2, BigDecimal.ROUND_FLOOR));//1.12
System.out.println(b1.setScale(3, BigDecimal.ROUND_FLOOR));//1.120

BigDecimal b2 = b1.negate();
System.out.println(b2.setScale(2, BigDecimal.ROUND_FLOOR));//-1.13
System.out.println(b2.setScale(3, BigDecimal.ROUND_FLOOR));//-1.121
复制代码

注意,此舍入模式始终不会增加计算值。

  • BigDecimal.ROUND_UNNECESSARY 断言请求的操作具有精确的结果,因此不需要舍入。 如果对获得精确结果的操作指定此舍入模式,则抛出ArithmeticException。
BigDecimal b1 = new BigDecimal("1.25");
System.out.println(b1.setScale(2, BigDecimal.ROUND_UNNECESSARY));//1.25
System.out.println(b1.setScale(1, BigDecimal.ROUND_UNNECESSARY));//throw ArithmeticException
复制代码

三、与之相关的两个类

  1. java.math.MathContext 该对象是封装上下文设置的不可变对象,它描述数字运算符的某些规则,如数据的精度,舍入方式等
    • MathContext.DECIMAL32 一个 MathContext 对象,其精度设置与 IEEE 754R Decimal32 格式(即 7 个数字)匹配,舍入模式为 HALF_EVEN,这是 IEEE 754R 的默认舍入模式。

    • MathContext.DECIMAL64 一个 MathContext 对象,其精度设置与 IEEE 754R Decimal64 格式(即 16 个数字)匹配,舍入模式为 HALF_EVEN,这是 IEEE 754R 的默认舍入模式。

    • MathContext.DECIMAL128 一个 MathContext 对象,其精度设置与 IEEE 754R Decimal128 格式(即 34 个数字)匹配,舍入模式为 HALF_EVEN,这是 IEEE 754R 的默认舍入模式。

    • MathContext.UNLIMITED 其设置具有无限精度算法所需值的 MathContext 对象。

  2. java.math.RoundingMode 枚举类,定义常用的数据舍入方式
    • HALF_UP(BigDecimal.ROUND_HALF_UP)
    • HALF_DOWN(BigDecimal.ROUND_HALF_DOWN)
    • HALF_EVEN(BigDecimal.ROUND_HALF_EVEN)
    • UP(BigDecimal.ROUND_UP)
    • DOWN(BigDecimal.ROUND_DOWN)
    • CEILING(BigDecimal.ROUND_CEILING)
    • FLOOR(BigDecimal.ROUND_FLOOR)
    • UNNECESSARY(BigDecimal.ROUND_UNNECESSARY)

【注】详见第二部分:舍入方式

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java BigDecimalJava中用于高精度计算的类。它允许我们进行任意精度的数字计算,而不会出现舍入误差。 在Java中,基本数据类型(如int、double等)的计算是有限制的。例如,double类型只能存储15位有效数字,而且在计算过程中可能会出现舍入误差。这在需要精确计算的场合下是不可接受的,这就需要使用BigDecimal类。 以下是一些Java BigDecimal的常用方法: 1. 实例化BigDecimal对象 可以使用BigDecimal的构造函数来实例化一个对象,例如: ``` BigDecimal num1 = new BigDecimal("1234.5678"); BigDecimal num2 = new BigDecimal(9876.5432); ``` 2. 加法、减法、乘法和除法 可以使用add()、subtract()、multiply()和divide()方法进行加、减、乘和除运算,例如: ``` BigDecimal result1 = num1.add(num2); BigDecimal result2 = num1.subtract(num2); BigDecimal result3 = num1.multiply(num2); BigDecimal result4 = num1.divide(num2, 2, RoundingMode.HALF_UP); // 保留两位小数 ``` 3. 取反、取绝对值、取反余弦等 可以使用negate()、abs()、acos()等方法进行相应的计算,例如: ``` BigDecimal result5 = num1.negate(); // 取反 BigDecimal result6 = num1.abs(); // 取绝对值 BigDecimal result7 = new BigDecimal(Math.PI).acos(); // 取反余弦 ``` 4. 比较大小 可以使用compareTo()方法进行大小比较,例如: ``` int cmp = num1.compareTo(num2); if (cmp > 0) { System.out.println("num1 > num2"); } else if (cmp < 0) { System.out.println("num1 < num2"); } else { System.out.println("num1 = num2"); } ``` 以上是Java BigDecimal的一些常用方法,使用BigDecimal类可以很方便地进行高精度计算。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值