【JAVA入门】Day19 - BigInteger 和 BigDecimal

【JAVA入门】Day19 - BigInteger 和 BigDecimal



        在 Java 中,整数有四种类型:byte,short,int,long。他们在底层分别占用的字节数如下:

类型占用字节数(byte)
byte1
short2
int4
long8

        当我们想要表示的数字超出 long 的取值范围时,就需要用到 BigInteger 类。

一、BigInteger

1.1 BigInteger 构造方法

方法名说明
public BigInteger(int num, Random rnd)获取随机大整数,范围:[0 ~ 2的num次方-1]
public BigInteger(String val)获取指定的大整数
public BigInteger(String val, int radix)获取指定进制的大整数
public static BigInteger valueOf(long val)静态方法获取BigInteger的对象,内部有优化

        注意:BigInteger 对象一旦创建完毕,里面的数据不能发生改变。

package BigNum;

import java.math.BigInteger;
import java.util.Random;

public class BigIntegerDemo1 {
    public static void main(String[] args) {
        //1.获取一个随机的的大整数
        Random r = new Random();
        BigInteger bd1 = new BigInteger(4, r);     //[0,2^4-1]
        System.out.println(bd1);

        //2.获取一个指定的大整数
        //字符串中,必须是整数,否则会报错
        BigInteger bd2 = new BigInteger("99999999999999999999999999999");
        System.out.println(bd2);

        //3.获取一个指定进制的大整数
        //生成二进制的100,输出时会自动转换为十进制
        //字符串中的数字必须是整数,字符串中的数字必须要和进制吻合,二进制中只能写0和1
        BigInteger bd3 = new BigInteger("100", 2);
        System.out.println(bd3);

        //4.静态方法获取BigInteger的对象,内部有优化
        //能表示的范围比较小,只能在long的取值范围内
        //在内部对常用的数字:-16 ~ 16 进行了优化
        //会提前把 -16 ~ 16 先创建好BigInteger的对象,如果多次获取不会再创建新的对象
        BigInteger bd4 = BigInteger.valueOf(100);
        System.out.println(bd4);

        //验证
        BigInteger bd5 = BigInteger.valueOf(16);
        BigInteger bd6 = BigInteger.valueOf(16);

        System.out.println(bd5 == bd6); //true

        BigInteger bd7 = BigInteger.valueOf(17);
        BigInteger bd8 = BigInteger.valueOf(17);

        System.out.println(bd7 == bd8); //false

        //5.对象一旦创建,内部的数据不能发生改变
        BigInteger bd9 = BigInteger.valueOf(1);
        BigInteger bd10 = BigInteger.valueOf(2);
        BigInteger result = bd9.add(bd10);
        System.out.println(result);    //3
        //add方法把bd9和bd10加起来,此时,不会修改参与计算的BigInteger对象中的值,而是产生了一个新的BigInteger的对象来记录3
        
    }
}

        如果你要创建的数据比较小,建议直接使用静态方法 public static BigInteger valueOf(long val) 来创建对象,会节约内存。

1.2 BigInteger 内部常见方法

在这里插入图片描述

        BigInteger 作为一个对象,是不能直接用基本运算符进行操作的,所有的数学运算,都需要用方法来进行。

package BigNum;

import java.math.BigInteger;

public class BigIntegerDemo2 {
    public static void main(String[] args) {
        //1.创建两个BigInteger对象
        BigInteger bd1 = BigInteger.valueOf(10);
        BigInteger bd2 = BigInteger.valueOf(5);

        //2.加法
        BigInteger bd3 = bd1.add(bd2);
        System.out.println(bd3);

        //3.减法
        BigInteger bd4 = bd1.subtract(bd2);
        System.out.println(bd4);

        //4.除法得到商和余数
        BigInteger[] arr = bd1.divideAndRemainder(bd2);
        System.out.println(arr[0]);
        System.out.println(arr[1]);

        //5.判断两个对象内部数据是否相同
        boolean result = bd1.equals(bd2);
        System.out.println(result);

        //6.次幂
        BigInteger bd5 = bd1.pow(2);

        //7.max
        //max或min方法会直接把较大的那个对象本身返回,并没有创建新的对象
        BigInteger bd6 = bd1.max(bd2);
        System.out.println(bd6 == bd1);     //true
        System.out.println(bd6 == bd2);     //false

        //8.转为int,超出int范围则报错
        BigInteger bd7 = BigInteger.valueOf(2147483647);
        int i  = bd7.intValue();
        System.out.println(i);              //2147483647

        //9.转为long
        BigInteger bd8 = BigInteger.valueOf(2147483648L);
        long l = bd8.longValue();
        System.out.println(l);              //2147483648

        //10.转为double
        BigInteger bd9 = BigInteger.valueOf(100);
        double d = bd9.doubleValue();
        System.out.println(d);              //100.0

    }
}

二、BigDecimal

        计算机中的小数由于存储的方式,导致做运算会出现结果不精确的情况。
在这里插入图片描述
        二进制存储的原理是 2 的次幂和相加,一些不是 2 的倍数的部分就会拆分成多个 2 的次幂相加的形式。如果我们把十进制的小数转换为二进制,后面的小数部分可能变得非常非常长。
        在 Java 中,小数数据类型 float 和 double 的占用字节数是固定的。float 占用 4个字节,也就是 32个bit位;double 占用8个字节,也就是 64个bit位。这些bit位中,还要腾出给整数部分存储的位数,留给小数部分的位数就变成了:float 剩余 23个bit位,double 剩余 52个bit位。超出的部分,只能舍弃。
        上图中,0.226 转换为二进制后,小数部分长达 55 位,已经超过了 double 能装下的最大位数,因此只能舍弃最后3位,因此该数据在计算机中的存储,将变得不精确(失真)
        但是我们在生产生活中,往往需要将小数精确表达,为此,Java 提供了一个精确的小数API——BigDecimal。

2.1 BigDecimal 的作用

  • 存储精确的浮点数
  • 进行浮点数的各种精确操作运算

2.2 BigDecimal 对象的获取

        可以使用构造方法或静态方法获取,用途有别。

package BigNum;

import java.math.BigDecimal;

public class BigDecimalDemo1 {
    public static void main(String[] args) {
        //1.通过传递double类型的小数来创建对象
        //细节:这种方式有可能是不精确的,不建议使用
        BigDecimal bd1 = new BigDecimal(0.01);      //0.01000000000000000020816681711721685132943093776702880859375
        BigDecimal bd2 = new BigDecimal(0.09);      //0.0899999999999999966693309261245303787291049957275390625

        System.out.println(bd1);
        System.out.println(bd2);

        //2.通过传递字符串表示的小数来创建对象
        BigDecimal bd3 = new BigDecimal("0.01");    //0.01
        BigDecimal bd4 = new BigDecimal("0.09");    //0.09

        System.out.println(bd3);
        System.out.println(bd4);

        //3.进行求和运算
        BigDecimal bd5 = bd3.add(bd4);
        System.out.println(bd5);                        //0.10

        //4.通过静态方法获取对象
        BigDecimal bd6 = BigDecimal.valueOf(10);
        System.out.println(bd6);                        //10

        //细节:
        //1.如果要表示的数字不大,没有超出double的取值范围,建议使用静态方法BigDecimal.valueOf()
        //2.如果要表示的数字比较大,超出了double的取值范围,建议使用构造方法
        //3.如果我们传递的是0~10之间的整数,那么方法会返回已经创建好的对象,不会重新new

        BigDecimal bd7 = BigDecimal.valueOf(10);
        System.out.println(bd6 == bd7);                 //true
        BigDecimal bd8 = BigDecimal.valueOf(10.0);
        System.out.println(bd6 == bd8);                 //false  如果传递double类型,一定会创建新的对象
        
    }
}

2.3 BigDecimal 中常见的成员方法

在这里插入图片描述

package BigNum;

import java.math.BigDecimal;
import java.math.RoundingMode;

public class BigDecimalDemo2 {
    public static void main(String[] args) {
        //1.加法
        BigDecimal bd1 = BigDecimal.valueOf(10.0);
        BigDecimal bd2 = BigDecimal.valueOf(2.0);
        BigDecimal bd3 = bd1.add(bd2);
        System.out.println(bd3);

        //2.减法
        BigDecimal bd4 = bd1.subtract(bd2);
        System.out.println(bd4);

        //3.乘法
        BigDecimal bd5 = bd1.multiply(bd2);
        System.out.println(bd5);

        //4.除法
        BigDecimal bd6 = bd1.divide(bd2);       //如果能除尽,可以用这行方法
        BigDecimal bd7 = bd1.divide(bd2, 2, RoundingMode.HALF_UP);      //如果除不尽,必须用这行方法

        //舍入模式
        //UP:向远离零的方向舍入
        //DOWN:向零的方向舍入
        //CEILING:向正无限大方向舍入
        //FLOOR:向负无限大方向舍入
        //HALF_UP:向最接近数字方向舍入(四舍五入)
        //HALF_DOWN:向最近数字方向舍入(四舍五舍)
        
    }
}

2.4 BigDecimal 的底层存储方式

        实际上 BigDecimal 在底层是一个数组,它是这样存储的:

BigDecimal bd = new BigDecimal("0.226");

        它会将"0.226"拆分成’0’,‘.’,‘2’,‘2’,‘6’,这五个字符,然后寻找他们对应的 ASC-II 码,存储在一个 byte 类型的数组中。

['0','.','2','2','6']
[48, 46, 50, 50, 54]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值