有关JAVA自动装箱-拆箱的分析

43 阅读2分钟

携手创作,共同成长!这是我参与「掘金日新计划 · 8 月更文挑战」的第24天,点击查看活动详情

@[toc] 在java常量与常量池 中已经介绍过一些java自动装箱与拆箱的例子。现在单独对自动装箱/拆箱进行总结。

1.JAVA数据类型

image.png

java基本数据类型和其包装类对应如下:

基础数据类型长度包装类型
int4字节Integer
byte1字节Byte
short2字节Short
long8字节Long
float4字节Float
double8字节Double
char2字节Character
boolean不同虚拟机取值不同Boolean

2.定义

自动装箱:把基本类型用其包装类替代,使其具有对象的特性。可以调用toString()、hashCode()、getClass()、equals()等方法。 如:

Integer a = 4;

其实编译器调用的是static Integer valueOf(int i)这个方法,valueOf(int i)返回一个表示指定int值的Integer对象,反编译后如下:

Integer a =Integer.valueOf(4);

自动拆箱:将包装类转换为基本类型。 如:

int a = new Integer(4);

实际上编译器内部会调用int intValue()返回该Integer对象的int值。 反编译后如下:

int a = new Integer(4).intValue();

3.发生时机

当出现赋值运算、算数表达式、方法调用等情况时,会触发自动装箱/拆箱操作。

 Integer a = 1;
 Integer b = 2;
 Long c = 3L;
 System.out.println(c == (a + b));
 System.out.println(c.equals(a + b));

上述问题输出结果:

true
false

说明: c==(a+b),算数表达式先运算 a+b 拆箱操作,得到数值为3。Long与int比较,会自动拆箱,因此最终是 3==3 ,得到值为true。 c.equals(a+b), a+b 拆箱操作,得到数值为 3 int类型。Long与int进行equals,由于不是同一个对象,因此会返回false。 可以查看 Long对象源码,equals 首先需要判断 instanceof 是否为同一对象。

 public boolean equals(Object obj) {
        if (obj instanceof Long) {
            return value == ((Long)obj).longValue();
        }
        return false;
    }

4.包装类缓存

查看包装类源码,对于 int byte short long char boolean 这些类型的包装类都实现了一个字节的缓存,(Boolean 只有true和false两种)。float double这两种类型没有缓存。因此在缓存范围内的包装类,使用==时实际是同一对象。

Integer i1 = 127;
Integer i2 = 127;
Integer i3 = 128;
Integer i4 = 128;
System.out.println(i1 == i2); //true  满足缓存访问
System.out.println(i3 == i4);//false 在缓存范围之外

Double d1 = 127.0;
Double d2 = 127.0;
Double d3 = 128.0;
Double d4 = 128.0;
System.out.println(d1 == d2);//false 没有缓存
System.out.println(d3 == d4);//false 没有缓存