BigDecimal 类型的使用

目录

一、前言

二、BigDecimal构造方法

二、BigDecimal参与运算

2.1定义初始值

2.2计算

2.3比较大小

2.4BigDecimal取其中最大、最小值、绝对值、相反数:

2.5补充

2.6、java中 BigDecimal类型的可以转换到double类型:

三、BigDecimal格式化、小数点转换

四、总结:


一、前言

float和double类型的主要设计目标是为了科学计算和工程计算,它们没有提供完全精确的结果,所以不应该被用于要求精确结果的场合。商业计算往往要求结果精确,这时候BigDecimal就派上大用场啦。所以项目与金融挂钩的小伙伴们应该是比较熟悉的。

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

二、BigDecimal构造方法

 1.public BigDecimal(double val)    将double表示形式转换为BigDecimal *不建议使用

 2.public BigDecimal(int val)  将int表示形式转换成BigDecimal

 3.public BigDecimal(String val)  将String表示形式转换成BigDecimal

public static void main(String[] args)
    {
        BigDecimal bigDecimal = new BigDecimal(2);
        BigDecimal bDouble = new BigDecimal(2.3);
        BigDecimal bString = new BigDecimal("2.3");
        System.out.println("bigDecimal=" + bigDecimal);
        System.out.println("bDouble=" + bDouble);
        System.out.println("bString=" + bString);
    }
 
bDouble结果会有精度的问题
为什么会出现这种情况呢?
 
 JDK的描述:1、参数类型为double的构造方法的结果有一定的不可预知性。有人可能认为在Java中写入newBigDecimal(0.1)所创建的BigDecimal正好等于 0.1(非标度值 1,其标度为 1),但是它实际上等于0.1000000000000000055511151231257827021181583404541015625。这是因为0.1无法准确地表示为 double(或者说对于该情况,不能表示为任何有限长度的二进制小数)。这样,传入到构造方法的值不会正好等于 0.1(虽然表面上等于该值)。
 
        2、另一方面,String 构造方法是完全可预知的:写入 newBigDecimal("0.1") 将创建一个 BigDecimal,它正好等于预期的 0.1。因此,比较而言,通常建议优先使用String构造方法。
 
 
 
当double必须用作BigDecimal的源时,请使用Double.toString(double)转成String,然后使用String构造方法,或使用BigDecimal的静态方法valueOf
public static void main(String[] args)
    {
        BigDecimal bDouble1 = BigDecimal.valueOf(2.3);
        BigDecimal bDouble2 = new BigDecimal(Double.toString(2.3));
 
        System.out.println("bDouble1=" + bDouble1);
        System.out.println("bDouble2=" + bDouble2);
        
    }

二、BigDecimal参与运算

2.1定义初始值

注:除BigDecimal.ZERO之外还有其他常量可以定义初始值

BigDecimal b1=new BigDecimal(0);
BigDecimal b2 = BigDecimal.ZERO;

2.2计算

加法

BigDecimal add(BigDecimal val) //BigDecimal 加法

减法:

BigDecimal subtract (BigDecimal val) //BigDecimal 减法

乘法:

BigDecimal multiply (BigDecimal val)  //BigDecimal 乘法

除法:

BigDecimal divide (BigDecimal val,RoundingMode mode)  除法

2.3比较大小

可以通过BigDecimal的compareTo方法来进行比较。
返回的结果是int类型,-1表示小于,0是等于,1是大于。
 
看下面这个例子:
BigDecimal a = new BigDecimal("1.00");
BigDecmial b = new BigDecimal(1);
 
想比较一下a和b的大小,一般都会用equals
 
System.out.println(a.equals(b));
但是输出结果是:false
原因是:BigDecimal比较时,不仅比较值,而且还比较精度?
 
 
if(a.compareTo(b)==0) 结果是true
 
比较大小可以用 a.compareTo(b)
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.4BigDecimal取其中最大、最小值、绝对值、相反数:

 
  a.max (b) //比较取最大值
 
  a.min(b) //比较取最小值
 
  a.abs()//取最绝对值
 
  a.negate()//取相反数

2.5补充

BigDecimal枚举常量用法摘要  :
CEILING  :向正无限大方向舍入的舍入模式。
DOWN  :向零方向舍入的舍入模式。
FLOOR  :向负无限大方向舍入的舍入模式。
HALF_DOWN  :向最接近数字方向舍入的舍入模式,如果与两个相邻数字的距离相等,则向下舍入。
HALF_EVEN  :向最接近数字方向舍入的舍入模式,如果与两个相邻数字的距离相等,则向相邻的偶数舍入。
HALF_UP  :向最接近数字方向舍入的舍入模式,如果与两个相邻数字的距离相等,则向上舍入。
UNNECESSARY  :用于断言请求的操作具有精确结果的舍入模式,因此不需要舍入。
UP  : 远离零方向舍入的舍入模式。


2.6、java中 BigDecimal类型的可以转换到double类型:

用 变量.doubleValue();函数  即可将 BigDecimal 类型数据 转化为 double类型
 BigDecimal b2 = BigDecimal.ZERO;
 double v = b2.doubleValue();

三、BigDecimal格式化、小数点转换

BigDecimal可以与DecimalFormat结合使用,从而对金额格式化,如小数点后面统一保留两位,不够两位的补零,多余两位的舍入

public class Test {
    public static void main(String[] s){
        System.out.println(formatToNumber(new BigDecimal("12333.435")));
        System.out.println(formatToNumber(new BigDecimal(0)));
        System.out.println(formatToNumber(new BigDecimal("0.00")));
        System.out.println(formatToNumber(new BigDecimal("0.001")));
        System.out.println(formatToNumber(new BigDecimal("0.006")));
        System.out.println(formatToNumber(new BigDecimal("0.206")));
        System.out.println(formatToNumber(new BigDecimal("1.22")));
    }
    /**
     * @desc
     * @param obj 传入的小数
     * @return
     */
    public static String formatToNumber(BigDecimal obj) {
        // DecimalFormat默认使用的是进位方式是RoundingMode.HALF_EVEN,此舍入模式也称为“银行家算法”,主要在美国使用。
        //银行家算法:四舍六入五考虑,五后非零就进一,五后为零看奇偶,五前为偶应舍去,五前为奇要进一
        DecimalFormat df = new DecimalFormat("###,##0.00"); 
        return df.format(obj);
    }
}

执行结果:

BigDecimal 类型的使用_第1张图片

四、总结:

 (1)商业计算使用BigDecimal。

 (2)尽量使用参数类型为String的构造函数。

 (3) BigDecimal都是不可变的(immutable)的,在进行每一步运算时,都会产生一个新的对  象,所以在做加减乘除运算时千万要保存操作后的值。

(4)DecimalFormat的默认进位方式不是四舍五入,所以当小数点后面需要舍去的时候,肯能跟预想的不一样,具体可参考《关于DecimalFormat的取舍问题,DecimalFormat四舍五入的坑》

(5)new DecimalFormat(“###,##0.00”)小数点前面需要有个0,这样0-1之间的数字才会正常格式化;若##0.00的小数点前面没有0,则0-1之间的数字会被丢失掉小数点前的0。

你可能感兴趣的:(java)