好好学Java:从零基础到项目实战
上QQ阅读APP看书,第一时间看更新

4.3.1 大整数BigInteger

早期的编程语言为了节约计算机的内存,给数字变量定义了各种存储规格的数值类型,比如字节型byte只占用1字节大小,短整型short占用2字节大小,整型int占用4字节大小,长整型long占用8字节大小。但是长整型只能表达到-263~263-1,超出这个范围的巨大整数竟连long类型也放不下。何况现在无论是手机还是计算机的内存都是以GB计量的,因此原先锱铢计较几字节的数值类型便不合时宜了。为此,Java又设计了一种大整数类型BigInteger。BigInteger能够表示任意大小的整数,而不再局限于多少位的数值范围。

乍看起来,BigInteger仿佛与Integer相似,仅仅在类型开头添加了Big字样。事实上,它们的类型设计有颇多异曲同工之处,二者的很多基本方法是一模一样的,例如初始化赋值的valueOf方法、比较相等的equals方法以及转换为基本数字类型的几个方法(包括byteValue、shortValue、intValue、longValue、floatValue、doubleValue等)。当然,BigInteger要处理的可是超大整数,故而它的用法还是与Integer有所区别的。接下来一一介绍BigInteger特别的地方。

首先,介绍如何初始化一个大整数变量。前面介绍Integer的时候,提到Java代码有3种给包装变量赋值的方式,分别是使用等号直接赋予具体数字、调用valueOf方法赋值、通过关键字new创建指定数字的包装变量。然而到了大整数BigInteger这里,3种方式只剩下valueOf方法能够对大整数变量初始化。

其次,包装变量允许使用“+”“-”“*”“/”“%”等运算符执行四则运算,到了大整数变量这里却不能使用算术运算符,而要通过专门的计算方法才能开展运算。具体说来,大整数类型使用add方法取代了加法运算符“+”,使用subtract方法取代了减法运算符“-”,使用multiply方法取代了乘法运算符“*”,使用divide方法取代了除法运算符“/”,使用remainder方法取代了取余数运算符“%”,使用negate方法取代了负号运算符“-”。这些新方法的调用代码示例如下(完整代码见本章源码的src\com\method\big\TestInteger.java):

        BigInteger nine=BigInteger.valueOf(9);      //生成一个指定数值的大整数变量
        BigInteger four=BigInteger.valueOf(4);      //生成一个指定数值的大整数变量
        BigInteger sum=nine.add(four);              //add方法用来替代加法运算符“+”
        System.out.println("sum="+sum);
        BigInteger sub=nine.subtract(four);      //subtract方法用来替代减法运算符“-”
        System.out.println("sub="+sub);
        BigInteger mul=nine.multiply(four);      //multiply方法用来替代乘法运算符“*”
        System.out.println("mul="+mul);
        BigInteger div=nine.divide(four);          //divide方法用来替代除法运算符“/”
        System.out.println("div="+div);
        BigInteger remainder=nine.remainder(four);  //remainder方法用来替代取余数运算符“%”
        System.out.println("remainder="+remainder);
        BigInteger neg=nine.negate();              //negate方法用来替代负号运算符“-”
        System.out.println("neg="+neg);

再次,Java虽然提供了常用的数学函数库Math,但是Math库只能操作基本数字类型的变量,不能操作大数字类型的变量。因而BigInteger另外提供了abs方法和pow方法,分别用于求大数字变量的绝对值和大数字变量的N次方。下面是大整数类型BigInteger调用这两个方法的代码例子:

        BigInteger abs=nine.abs();  // abs方法用来替代数学库函数Math.abs
        System.out.println("abs="+abs);
        BigInteger pow=nine.pow(2);  // pow方法用来替代数学库函数Math.pow
        System.out.println("pow="+pow);

总结一下,包装数字类型相比基本数字类型,表达的数值范围并没有扩大,仅仅是调用方式上有所区别,可谓是换汤不换药。而大数字类型真正解决了数值范围的表达限制,并且取消了带有明显数学印记的算术运算符,这才形成了面向方法而非面向运算的编程风格。