美文网首页
JAVA之自动拆装箱

JAVA之自动拆装箱

作者: 煮黑豆 | 来源:发表于2019-06-21 14:48 被阅读0次

    自动拆装箱:就是8种基本类型与其对应包装器类型可以直接互换!

    注意:基本类型的数组,与包装器类型的数组不能互换!

    自动拆装箱是JDK5.0的新特性之一,这一特性可以使基本类型,与之对应的包装器类型之间直接转换,例如int的包装器类型是Integer!在JDK5.0之后,你甚至可以把int当作成Integer来使用,把Integer当成int来使用。当然,这不是100%的!

    自动拆装箱概述

    在JDK5.0之后,Java允许把基本类型与其对应的包装器类型之间自动相互转换。

    自动装箱:Integer i = 100,把int类型直接赋值给Integer类型;

    自动拆装:int a = new Integer(100),把Integer类型直接赋值给int类型。

    自动拆装箱原理

    其实自动拆装箱是由编译器完成的!我们写的代码,再由编译器“二次加工”,然后再编译成.class文件!那么编译器是怎么“二次加工”的呢?

    Integer i = 100:编译器加工为:Integer i = Integer.valueOf(100);

    int a = i:编译器加载为:int a = i.intValue();

    这也说明一个道理:JVM并不知道什么是自动拆装箱,JVM还是原来的JVM(JDK1.4之前),只是编译器在JDK5.0时“强大”了!

    自动拆装箱演变

    大家来看看下面代码:

    Integer i = 100;//这是自动装箱
    
    Object o = i;//这是身上转型
    
    

    上面代码是没有问题的,我们是否可以修改上面代码为:

    Object o=100;
    

    ok,这是可以的!通过编译器的处理后上面代码为:

    Object o = Integer.valueOf(100);
    

    在来看下面代码:

    Object o = Integer.valueOf(100);
    int a = o;//编译失败!
    

    上面代码是不行的,因为o不是Integer类型,不能自动拆箱,你需要先把o转换成Integer类型后,才能赋值给int类型。

    Object o = Integer.valueOf(100);
    int a = (Integer)o;
    

    自动拆箱装箱的误区
    来看下面代码:

    int[] intArr = {1,2,3};
    Integer[] integerArr = intArr;//编译失败
    

    上面代码无论编译通过!因为int[]是引用类型,而不是基本类型。而自动拆装箱只是在基本类型与其对应的包装器类型之间进行转换!也就是说,int[]和Integer[]是两种不同的引用类型。
    你可能会认为,把上面代码修改下面的样子即可,这说明需要还是没有理解这一概念。把int[]转换成Integer[],等同与把Integer转换成String类型一样,是不行的!所以下面代码还是编译失败。

    int[] intArr = {1,2,3};
    Integer[] integerArr = (Integer[])intArr;
    

    如果可以把int当成Integer来对待,那么是否可以:int a = null?当然是不行的,所以希望对自动拆装箱正确理解,它不是万能的。

    变态小题目

    Integer i1 = 100;
    Integer i2 = 100;
    boolean b1 = i1 == i2;//结果为true
    
    Integer i3 = 200;
    Integer i4 = 200;
    boolean b2 = i3 == i4;//结果为false
    

    你可能对上面代码的结果感到费解,那么我们来打开这个疑团。第一步,我们先把上面代码通过编译器的“二次加工”处理一下:

    Integer i1 = Integer.valueOf(100);
    Integer i2 = Integer.valueOf(100);
    boolean b1 = i1 == i2;//结果为true
    
    Integer i3 = Integer.valueOf(200);
    Integer i4 = Integer.valueOf(200);
    boolean b2 = i3 == i4;//结果为false
    

    这时你应该可以看到,疑团在Integer.valueOf()方法身上。传递给这个方法100时,它返回的Integer对象是同一个对象,而传递给这个方法200时,返回的却是不同的对象。这是我们需要打开Integer的源码(这里就不粘贴Integer的源代码了),查看它的valueOf()方法内容。
    Integer类的内部缓存了-128~127之间的256个Integer对象,如果valueOf()方法需要把这个范围之内的整数转换成Integer对象时,valueOf()方法不会去new对象,而是从缓存中直接获取,这就会导致valueOf(100)两次,都是从缓存中获取的同一个Integer对象!如果valueOf()方法收到的参数不在缓存范围之内,那么valueOf()方法会new一个新对象!这就是为什么Integer.valueOf(200)两次返回的对象不同的原因了。

    相关文章

      网友评论

          本文标题:JAVA之自动拆装箱

          本文链接:https://www.haomeiwen.com/subject/dybyqctx.html