什么是自动装箱与拆箱
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的对象层次中。
基本数据类型 | 对应包装类 |
---|---|
byte | Byte |
short | Short |
int | Integer |
long | Long |
float | Float |
double | Double |
char | Character |
boolean | Boolean |
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.low
到java.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
欢迎任何形式的转载,但请务必注明出处。
限于本人水平,如果文章和代码有表述不当之处,还请不吝赐教。
本文章仅作为自学所用。