Java 自动装箱的陷阱

Java 自动装箱的陷阱

1. 自动装箱

 自动装箱、拆箱是Java语言中使用的最多的语法糖之一。装箱就是自动将基本数据类型转换为包装器类型;拆箱就是自动将包装器类型转换为基本数据类型。

public class Main {
    public static void main(String[] args) {
    //自动装箱
    Integer total = 99;
    //自定拆箱
    int totalprim = total;
    }
}

 下面看一下需要装箱拆箱的类型有哪些:
java-auto-boxing-type

2. 自动装箱的过程

 下面我们看一下自动装箱、拆箱的具体过程。执行下面语句,查看 main 函数的字节码

javac AutoBoxing.java
javap -verbose AutoBoxing

 下面是 main 函数的字节码:

public static void main(java.lang.String[]);
    descriptor: ([Ljava/lang/String;)V
            flags: ACC_PUBLIC, ACC_STATIC
            Code:
            stack=1, locals=3, args_size=1
            0: bipush        99
            2: invokestatic  #2                  // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
            5: astore_1
            6: aload_1
            7: invokevirtual #3                  // Method java/lang/Integer.intValue:()I
            10: istore_2
            11: return
            LineNumberTable:
            line 7: 0
            line 9: 6
            line 10: 11

 bipush 将 99 放到栈顶,接下来使用 invokestatic,这里会调用了 Integer.valueOf 的静态方法,栈顶的数据作为参数,即这里实际执行的是 Integer.valueOf(99) 语句,这就是自动装箱的过程。接下来 astore_1 将 total 的引用放到栈顶,aload_1 将栈顶total的引用加载到了 slot1,接下来 invokevirtual 会调用 Integer.intValue 方法,即 total.intValue() 方法,istore_2 将该值放到了 slot2,即赋值给了totalprim,这里就完成了拆箱的过程。

3. 自动装箱的陷阱

 自动装箱、拆箱这些语法糖看似很简单,但这里面有很多我们需要注意的地方,下面代码演示了自动装箱的一些错误用法:

public static void main(String[] args){
        Integer a = 1;  
        Integer b = 2;  
        Integer c = 3;  
        Integer d = 3;  
        Integer e = 128; 
        Integer f = 128;  
        Long g = 3L;
        System.out.println(c == d); //true  
        System.out.println(e == f); //false  
        System.out.println(c == (a+b)); //true  
        System.out.println(c.equals(a+b)); //true  
        System.out.println(g == (a+b)); //true 
        System.out.println(g.equals(a+b)); //false 
    }

 这样的答案可能会出乎很多人的意料,接下来一一分析。
 首先明确一下 “==” 和 equals 方法的用法。
 “==”:如果是基本数据类型,则直接对值进行比较,如果是引用数据类型,则是对他们的地址进行比较(但是只能比较相同类型的对象,或者比较父类对象和子类对象。类型不同的两个对象不能使用==)。
 equals:装箱类型重写了 equals 方法,比较规则为:如果两个对象的类型一致,并且内容一致,则返回true。以 Integer 为例:

//Integer.java
public boolean equals(Object obj) {
        if (obj instanceof Integer) {
            return value == ((Integer)obj).intValue();
        }
        return false;
    }

 由前面分析可知,自动装箱的时候实际调用了 Integer.valueOf 方法:

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

 IntegerCache 是 Integer 内部定义的一个私有静态内部类:

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使用一个内部静态类中的一个静态数组(Integer cache[])保存了-128-127范围内的数据,静态数组在类加载以后是存在方法区的,并不是什么常量池。在自动装箱的时候,首先判断要装箱的数字的范围,如果在-128-127的范围则直接返回缓存中已有的对象,否则new一个新的对象。其他的包装类也有类似的实现方式。
 需要注意几点:

  • 包装类的 == 运算在不遇到算术运算的情况下不会自动拆箱
  • 类之间 == 比较的是地址
  • 包装类的 equals 方法不处理数据转型的关系

 因此,我们有以下分析:

Integer a = 1;  //直接获取IntegerCache内部的对象
Integer b = 2;  //直接获取IntegerCache内部的对象
Integer c = 3;  //直接获取IntegerCache内部的对象
Integer d = 3;  //直接获取IntegerCache内部的对象
Integer e = 128; //new一个新的对象
Integer f = 128;  //new一个新的对象
Long g = 3L;
System.out.println(c == d); //true  是同一个对象

c 和 d 指向的是 IntegerCache 内部同一个对象,地址一样,== 比较的是地址。

System.out.println(e == f); //false   //128开始就false,不同的对象

e 和 f 大于127,都是 new 生成的新对象,值相同,但是对应不同的内存地址。

System.out.println(c == (a+b)); //true  //拆箱变成基本类型

(a+b) 触发自动拆箱,自动拆箱后是基本类型,它们的值是相等的。

System.out.println(c.equals(a+b)); //true  //Integer和Integer类型一致,数值也一样

equals 的参数是 Object,(a+b)会触发自动拆箱,结果变成基本类型,传进 equals 方法的时候会执行自动装箱,结果和 c 的值相同,而且类型相同,返回true。

System.out.println(g == (a+b)); //true  //拆箱子

(a+b) 触发自动拆箱,自动拆箱后是基本类型,它们的值是相等的。

System.out.println(g.equals(a+b)); //false  //Long和Integer类型不一样

(a+b)传进 equals 方法,自动装箱变成 Integer,值相同但是类型不同,返回false。

4. 总结

 鉴于包装类的“==”运算在不遇到算数运算的情况下不会自动拆箱,以及它们的equals方法不处理数据转型的关系,我们在实际编码的时候应该尽量避免这样使用自动装箱与拆箱。
 

参考:《深入理解Java虚拟机》

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值