Java之自动装箱与拆箱

本文详细介绍了Java中的自动装箱和拆箱机制,包括基本数据类型、包装类、Integer的IntegerCache机制以及装箱和拆箱的实现过程。通过案例分析解释了装箱、拆箱的原理,并探讨了频繁操作对性能的影响。最后,通过一道面试题加深了读者的理解。
摘要由CSDN通过智能技术生成

什么是自动装箱与拆箱

1,来由

Java是一种面向对象编程的语言,但他同时也提供了基本数据类型,提供基本数据类型是出于性能方面的考虑:因为使用对象来处理即使是最简单的计算,系统也销也比较大。

Java中的基本数据类型没有方法和属性,但是在特定场景下,我们必须要利用对象的相关属性,而包装类就是为了让基本数据类型拥有方法和属性,实现对象化交互。这就是我们要学习的自动装箱与拆箱。

2,java中的基本数据类型

java中有8种基本数据类型,分别为:

  • 数字类型(6种)

    • 整型

      • byte--------------------1字节= 8bit

      • short-------------------2字节=16bit

      • int-----------------------4字节= 32bit

      • long--------------------8字节= 64bit

    • 浮点型

      • float-------------------4字节=32bit
      • double----------------8字节=64bit
  • 字符类型(1种):char-----2字节=16bit

  • boolean类型(1种):boolean(类型有两个值:false和true,用来判定逻辑条件。整型值和布尔值之间不能进行相互转换。。boolean类型占用存储空间官方未定义

注:在Java中,所有的数值类型所占据的字节数量是不变的,其不会随着机器硬件的变化而改变,而且Java中没有任何无符号类型(unsigned)

3,类型封装器(包装类)

上面说的八种基本数据类型都分别都有对应的包装类,如下表:需要了解的时这些包装类提供了大量的方法,通过这些方法可以完全将基本类型集成到Java的对象层次中。

基本数据类型对应包装类
byteByte
shortShort
intInteger
longLong
floatFloat
doubleDouble
charCharacter
booleanBoolean

4,自动装箱与拆箱

4.1 自动装箱

装箱 :就是将基本数据类型用他们对应的包装类包装起来。如下:

Integer i = 10;

我们将上述代码反编译即得到:

Integer i = Integer.valueOf(10);

查看源码:

public static Integer valueOf(int i) {
        if (i >= IntegerCache.low && i <= IntegerCache.high)
            return IntegerCache.cache[i + (-IntegerCache.low)];
        return new Integer(i);
}
    private static class IntegerCache {
        static final int low = -128;
        static final int high;
        static final Integer cache[];

        static {
            // high value may be configured by property
            int h = 127;
            String integerCacheHighPropValue =
                sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
            if (integerCacheHighPropValue != null) {
                try {
                    int i = parseInt(integerCacheHighPropValue);
                    i = Math.max(i, 127);
                    // Maximum array size is Integer.MAX_VALUE
                    h = Math.min(i, Integer.MAX_VALUE - (-low) -1);
                } catch( NumberFormatException nfe) {
                    // If the property cannot be parsed into an int, ignore it.
                }
            }
            high = h;

            cache = new Integer[(high - low) + 1];
            int j = low;
            for(int k = 0; k < cache.length; k++)
                cache[k] = new Integer(j++);

            // range [-128, 127] must be interned (JLS7 5.1.7)
            assert IntegerCache.high >= 127;
        }

        private IntegerCache() {}
}

在Integer的valueof(int i)中,我们传递进去的参数是基本数据类型int型,在函数内部如上所示利用了Integer内部的一个私有静态类IntegerCache,该类静态初始化了一个包含了Integer.IntegerCache.lowjava.lang.Integer.IntegerCache.high的Integer数组。其中java.lang.Integer.IntegerCache.high的取值范围在[127~Integer.MAX_VALUE - (-low) -1]之间。

在该区间内所有的Integer.valueOf(int)函数返回的对象,是根据int值计算的偏移量,从数组Integer.IntegerCache.cache中获取,返回一个Integer对象。如果不在[IntegerCache.low~IntegerCache.high)之间,就会在JVM中分配内存new一个新的Integer对象。

因此,我们可以看出来装箱的时候其实就是自动调用Integer的valueOf(int)方法

4.1 自动拆箱

拆箱就是将包装类型转换为基本数据类型。如下:

Integer i = 10;//装箱
int n = i;   //拆箱

反编译得到:

Integer i = Integer.valueOf(10);
int n = i.intValue();

查看源码:

private final int value;
public int intValue() {
    return value;
}

可以看出,在拆箱的时候调用了Integer对象的intValue()方法。包装类在装箱时,将我们传入的int型参数value保存起来且私有不能被改变,在拆箱时利用intValue()方法返回int型的value值。

因此总结一下装箱和拆箱的实现过程就是:

装箱过程是通过调用包装类的valueOf方法实现的,而拆箱过程是通过调用包装类的 xxxValue方法实现的。(xxx代表对应的基本数据类型)。

注意:我们要了解的是,频繁的装箱拆箱的话,会增加内存的消耗,影响性能

5,经典案例分析

我们利用一道高频面试来进行巩固上面的知识点:

下面这段代码的输出结果是什么呢?

public class Main {
    public static void main(String[] args) {
         
        Integer i1 = 100;
        Integer i2 = 100;
        Integer i3 = 200;
        Integer i4 = 200;

        Integer one=new Integer(100);
        Integer two=new Integer(100);
        System.out.println(i1==i2);   //1
        System.out.println(i3==i4);   //2    
        System.out.println(one==two);  //3
        System.out.println(i1==100);   //4
        System.out.println(i1==one);   //5
        

    }

这里我们提醒一下:”==“用于比较引用和比较基本数据类型时具有不同的功能,具体如下:

  • 基本数据类型:比较的是他们的值是否相等,比如两个int类型的变量,比较的是变量的值是否一样。
  • 引用数据类型:比较的是引用的地址是否相同,比如说新建了两个User对象,比较的是两个User的地址是否一样。

解答:

1,true,上面的代码中i1和i2的数值为100,且被Integer包装类包装起来,因此直接从IntegerCache私有静态类的Integer数组中取已经存在的对象。

2,false,这里很多同学容易出错,上面的代码中i3和i4的数值为200,根据IntegerCache私有静态类的源码可以发现,当数值不在[127~Integer.MAX_VALUE - (-low) -1]即[-128,127]之间时,便会创建新的Integer对象。因此这里i3和i4是两个不同对象,因此为false。

3,false,这里就很好理解了,因为new Integer(100);创建了两个不同的对象,==比较的是内存地址。

4,true,这里其实包含两个步骤:1, Integer i1 = 100;进行装箱;2,i1==100自动进行了拆箱,因此这里是两个基本数据类型进行比较,值必然相同。

5,false,i1对象进行了装箱,其是直接从IntegerCache私有静态类的Integer数组中取已经存在的对象。而one是创建了一个新的对象,因此两者的内存地址必然不相同。

总结 : 如果我们单纯的看自动装箱与拆箱是很难理解的,当我们结合着实际的案例去学习就会事倍功半。

作者:王二黑_Leon

欢迎任何形式的转载,但请务必注明出处。

限于本人水平,如果文章和代码有表述不当之处,还请不吝赐教。

本文章仅作为自学所用。

评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

王小二_Leon

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值