JDK自动打包与解包机制(Auto boxing/unboxing)优化

【引言】

首先来看一段代码:
package test; /** * * @author whwang * 2011-12-3 上午11:44:04 */ public class Test { public static void main(String[] args) { Integer i1 = 100; Integer i2 = 100; System.err.println(i1 == i2); Integer i12 = 200; Integer i22 = 200; System.err.println(i12 == i22); } }
看上去“完全相同”的两部分,但一个输出true,一个false,为什么?

【解密】

首先需要知道,JDK的自动打包解包原理,打包是调用包装类的valueOf()方法,如上面的Integer i1 = 100等价于Integer i1 = Integer.valueOf(100),解包调用的对应的xxxValue(),如Integer调用intValue()方法解包。接下来我们看

看Integer.valueOf()这个方法,源码如下:
public static Integer valueOf(int i) { final int offset = 128; if (i >= -128 && i <= 127) { // must cache return IntegerCache.cache[i + offset]; } return new Integer(i); }
在该方法中,判断这个值是否在[-128,127]这个区间,如果是,则返回IntegerCache.cache[i + offset]。IntegerCache是Integer中的一个私有内部类,它的代码如下:
private static class IntegerCache { private IntegerCache() { } static final Integer cache[] = new Integer[-(-128) + 127 + 1]; static { for (int i = 0; i < cache.length; i++) cache[i] = new Integer(i - 128); } }
在IntegerCache中,静态创建值等于[-128,127]的整数的包装对象,并将它们缓存起来。再看看文章开始时提出的那段代码,System.err.println(i1 == i2)由于i1和i2的值为100,在[-128,127]之间,所以i1和i2返回的都是缓存的对象,两个对象完全相同,输出true。而i12和i22不在[-128,127]区间,在valueOf()方法中,不在该区间的整数,返回的return new Integer(i),所以i12和i22的地址也就不同了,输出false。类似的,Byte、Short、Long都与Integer类似,都有缓存数值为[-128,127]之间的对象。

这样做的好处显而易见,假如在我们的程序中大量的使用[-128,127]这区间的Integer对象,则不需要使用new来实例化对象,而是直接从缓存中拿,节省时间/空间。

【注意】

我们可以看到,在Byte、Short、Integer、Long这4个包装类中,都有3个重载的valueOf()方法,下面以Integer为例子:
Integer.valueOf(100); Integer.valueOf("100"); // 等价于Integer.valueOf("100", 10); Integer.valueOf("100", 10);而在这3个方法, 只有第一个Integer.valueOf(int)方法有利用缓存,其他两个方法都是直接new对象返回的。Byte、Short、Long与Integer类似.


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值