【JAVA入门】Day19 - BigInteger 和 BigDecimal
文章目录
- 【JAVA入门】Day19 - BigInteger 和 BigDecimal
- 一、BigInteger
- 1.1 BigInteger 构造方法
- 1.2 BigInteger 内部常见方法
 
- 二、BigDecimal
- 2.1 BigDecimal 的作用
- 2.2 BigDecimal 对象的获取
- 2.3 BigDecimal 中常见的成员方法
- 2.4 BigDecimal 的底层存储方式
 
 
在 Java 中,整数有四种类型:byte,short,int,long。他们在底层分别占用的字节数如下:
| 类型 | 占用字节数(byte) | 
|---|---|
| byte | 1 | 
| short | 2 | 
| int | 4 | 
| long | 8 | 
当我们想要表示的数字超出 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]



















