Java中BigDecimal类详解

前言:float和double只能用来做科学计算或者是工程计算。在商业计算中,对数字精度要求较高,必须使用 BigInteger 类和 BigDecimal 类,它支持任何精度的定点数,可以用它来精确计算货币值。

 BigDecimal本身支持基础的数学计算,可以使用BigDecimal还有一个非常重要的目的,可以利用它来实现准确的四舍五入操作。

1.BigDecimal构造器

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

以上都是比较常用的构造器,他们返回的对象都是BigDecimal对象。换而言之,将各个类型的值转换为BigDecimal对象,就是通过构造器。如果想将BigDecimal对象转换为其他类型的对象,我们通过以下几种:

toString()将BigDecimal对象的数值转换成字符串。
doubleValue()将BigDecimal对象中的值以双精度数返回。
floatValue()将BigDecimal对象中的值以单精度数返回。
longValue()将BigDecimal对象中的值以长整数返回。
intValue()将BigDecimal对象中的值以整数返回。

2.BigDecimal常用方法

注意,由于一般数值类型,例如double,不能准确地代表16位有效数以上的数字,在使用BigDecimal时,应用 BigDecimal(String)构造器创建对象才有意义。另外,BigDecimal所创建的是对象,我们不能使用传统的+、-、*、/等算术运算 符直接对其对象进行数学运算,而必须调用其相对应的方法。方法中的参数也必须是BigDecimal的对象。 

2.1 加减乘除

  1. 加法:add

  2. 减法:subtract

  3. 乘法:multiply
  4. 除法:divide

代码示例如下:

package com.example.pdf.test;

import java.math.BigDecimal;

public class Demo3 {

    public static void main(String[] args) {
        BigDecimal b1 = new BigDecimal("10");
        BigDecimal b2 = new BigDecimal("50");
        
        System.out.println("加法,求两个BigDecimal类型数据的和:"+b1.add(b2));
        System.out.println("减法,求两个BigDecimal类型数据的差:"+b1.subtract(b2));
        System.out.println("乘法,求两个BigDecimal类型数据的积:"+b1.multiply(b2));
        System.out.println("求余数,求b1除以b2的余数:"+b1.remainder(b2));
        System.out.println("最大数,求两个BigDecimal类型数据的最大值:"+b1.max(b2));
        System.out.println("最小数,求两个BigDecimal类型数据的最小值:"+b1.min(b2));
        System.out.println("绝对值,求BigDecimal类型数据的绝对值:"+b1.abs());;
        System.out.println("相反数,求BigDecimal类型数据的相反数:"+b1.negate());;


    }
}

结果如下所示:

除法 divide有三个参数的方法,第一参数表示除数,第二个参数表示小数点后保留位数,第三个参数表示取舍规则。

BigDecimal divide(BigDecimal divisor, int scale, int roundingMode)

只有在作除法运算或四舍五入时才用到取舍规则, 因为BigDecimal除法可能出现不能整除的情况,比如 4.5/1.3,这时会报错java.lang.ArithmeticException: Non-terminating decimal expansion; no exact representable decimal result。所以当我们用三参数的除法方法时,规定了保留几位小数以及你的保留方式,就可以避免异常。

我们最常用的四舍五入就是是 ROUND_HALF_UP,下面列几个比较常见的取舍规则:

ROUND_CEILING向正无穷方向舍入。
ROUND_DOWN向零方向舍入
ROUND_FLOOR向负无穷方向舍入
ROUND_HALF_DOWN向(距离)最近的一边舍入,除非两边(的距离)是相等,如果是这样,向下舍入, 例如1.55 保留一位小数结果为1.5
ROUND_HALF_EVEN向(距离)最近的一边舍入,除非两边(的距离)是相等,如果是这样,如果保留位数是奇数,使用ROUND_HALF_UP,如果是偶数,使用ROUND_HALF_DOWN
ROUND_HALF_UP向(距离)最近的一边舍入,除非两边(的距离)是相等,如果是这样,向上舍入, 1.55保留一位小数结果为1.6
ROUND_UNNECESSARY计算结果是精确的,不需要舍入模式
ROUND_UP向远离0的方向舍入

例如,RoundingMode其实是个枚举类,点进去源码可以看到其实他就是匹配到几种取舍规则

a.divide(b,2,RoundingMode.HALF_UP)

2.2 大小比较

package com.example.pdf.test;

import java.math.BigDecimal;

public class Demo3 {

    public static void main(String[] args) {
        BigDecimal a = new BigDecimal(20);
        BigDecimal b  = new BigDecimal(40);
        //前提为a、b均不能为null
        if(a.compareTo(b) == -1){
            System.out.println("a小于b");
        }

        if(a.compareTo(b) == 0){
            System.out.println("a等于b");
        }

        if(a.compareTo(b) == 1){
            System.out.println("a大于b");
        }

        if(a.compareTo(b) > -1){
            System.out.println("a大于等于b");
        }

        if(a.compareTo(b) < 1){
            System.out.println("a小于等于b");
        }

    }
}

2.3 四舍五入

setScale()有3个方法,第一个参数就是你要保留几位,第二个可填的参数就是取舍规则。如果你第二个参数不加,仅仅想保留几位,会自动帮你选择默认的规则。

ROUND_UNNECESSARY:计算结果是精确的,不需要舍入模式。

3.数据库设计

BigDecimal在进行入库时, 数据库选择decimal类型, 长度可以自定义,如10; 小数点我们项目中用的是4, 保留4位小数. 此外还要注意的就是默认值, 一定写成0.0000, 不要用默认的NULL, 否则在进行加减排序等操作时, 会带来转换的麻烦。

numericalValue decimal(10,2) DEFAULT '0.0000' COMMENT '数值',

 

--------------如果大家喜欢我的博客,可以点击左上角的关注哦。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值