Java包装类

尽管 Java 语言是典型的面向对象编程语言,但其中的八种基本数据类型并不支持面向对象编程,即基本类型的数据不具备“对象”的特性——不携带属性、没有方法可调用

Java同时又为每种基本数据类型分别设计了对应的类,称之为包装类。

基本数据类型 包装类 取值范围
byte Byte -128~127
short Short -32768~32767
int Integer -231 ~ 2(31-1)
long Long -263 ~ 2(63-1)
char Character 0~65535
float Float 1.4E-45~3.4028235E38
double Double 4.9E-324~1.7976931348623157E308
boolean Boolean true或false

每个包装类的对象可以封装一个对应的基本类型的数据,并提供了其它一些有用的方法。

为什么有基本类型还要有包装类型?

首先,基本类型并不具有对象的性质,为了让基本类型也具有对象的特征,就出现了包装类型(如我们在使用集合类型Collection时就一定要使用包装类型而非基本类型),它相当于将基本类型“包装起来”,使得它具有了对象的性质,并且为其添加了属性和方法,丰富了基本类型的操作。

另外,集合类(ArrayList,HashMap等)里面保存的必须是Object类型的对象,无法保存基本类型。

那只有包装类型不就可以了?

我们知道Java中,new一个对象会存储在堆里,我们通过栈中的引用来使用这些对象。由于基本数据类型很常用,如果我们用new将其存储在堆里效率明显很低,尤其是基本数据类型都很简单,仅仅只是一个变量而已。

关于默认值

整型byte、short、int、long的默认值都为0,浮点型float、double的默认值为0.0,boolean默认值为false,char默认值为空。对应的包装类型默认值都为null。

好多博客里都有这段话,但一旦自己动手实践过就会发现这里是有个坑的:
局部变量不会有默认值,必须要初始化才能使用。例如如下代码:

        int a;
        System.out.println(a);

会在第二行报错:

The local variable a may not have been initialized

只有当基本类型作为作为类成员使用时,Java则会确保给定其初始值

    public static void main(String[] args) {
        
        Person person = new Person();
        System.out.println(person.age);
    }
    
    static class Person{
        public int age;
    }
    

可以正常运行,输出结果为0。

这一功能主要是为了防止程序运行时错误。但是这些初始值对我们的程序来说绝大多数时候是不正确的。所以使用时必须明确指定初始值。

自动拆箱和装箱

Java从jdk1.5开始引入了自动装箱和拆箱,使得基本数据类型与包装类之间相互转换变得非常简单。

自动装箱: java自动将基本类型转化为包装类的过程,自动装箱时编译器会调用valueOf方法,将基本类型转化为包装类。

自动拆箱: java自动将包装类转化为基本类型的过程,自动拆箱时编译器会调用intValue(),doubleValue()这类的方法将对象转换成基本类型值。

自动装箱、拆箱带来的问题

由于装箱会隐式地创建对象,多余的对象会增加GC的压力,影响程序的性能。因此,最好不要在循环中使用自动装箱。

另外,如下代码:

    public static void main(String[] args) {
        Integer t1 = 128;
        Integer t2 = 128;
        System.out.println(t1 == t2);//false
        /**
         * 在编译阶段自动装箱后,代码变为
         * Integer t1 = new Integer(128);
         * Integer t2 = new Integer(128);
         * 会创建两个对象,因此不相等。
         */
        
        Integer t3 = 127;
        Integer t4 = 127;
        System.out.println(t3 == t4);//true
        /**
         * t3与t4之所以相等,是因为在JVM中有数字常量池,
         * 缓存的范围是-128~127,如果Integer指向这个范围内的数字,
         * 会在编译的时候会直接指向常量池中的数字,而不会创建新的对象
         */
    }

除Float和Double之外的基本类型在-128~127之间都会指向数组常量池。

你可能感兴趣的:(Java包装类)