首页 > 代码库 > 再学Java 之 Integer 包装类缓存

再学Java 之 Integer 包装类缓存

前言:本博文将涉及的Java的自动装箱和自动拆箱,可以参考 这篇文章 和 官方教程 ,这里不再赘述。

 

首先,先看一个小程序:

public class Main {
    
    public static void main(String[] args){
        Integer i1 = new Integer(1);
        Integer i2 = new Integer(1);
        System.out.println(i1 == i2);
        
        Integer i3 = 1;
        Integer i4 = 1;
        System.out.println(i3 == i4);
        
        Integer i5 = 200;
        Integer i6 = 200;
        System.out.println(i5 == i6);
    }
}

 

上面的程序会依次输出false 、true、false。

 第一个输出语句应该比较好理解,就是创建了不同的对象。但是第二跟第三个输出语句估计很多人就很难理解了。

要解释这个问题,需要从缓存说起。

 

缓存

  缓存是软件设计模式中一个非常有用的模式,缓存的实现方式有很多,不同方式可能存在性能上的差别。下面给出一个用数组实现的实例:

  (1)缓存类Cache_test

/*
 * <p>
 * 该对象使用数组实现了缓存,也就是,
 * 每一次使用valueOf()创建新对象时,系统将会确认缓存中是否已经存在相应的对象(即data相等)。
 * 假如存在,则直接返回缓存已存在的对象;
 * 假如不存在,则创建一个新对象,存储到缓存中,并返回新创建的对象。
 * </p>
 * 
 * @author Harvin.
 * @version 1.0
 */
public class Cache_test {
    //需要存储的数据
    private final String data;
    
    public Cache_test(String data){
        this.data =http://www.mamicode.com/ data;
    }
    public String get_data(){
        return this.data;
    }
    @Override
    //直接判断是否是指向同一个对象
    public boolean equals(Object obj){
        if (this == obj) {
            return true;
        }
        return false;
    }
    
    
    //定义缓存的大小
    private final static int MAX_SIZE = 10;
    //使用数组来存储缓存
    private static Cache_test[] cache
    = new Cache_test[MAX_SIZE];
    //定义当前缓存存储的位置
    private static int pos = 0;
    
    /* 判断是否已经缓存了包含该data对象的Cache_test对象,
     * 如果存在,则直接返回;
     * 如果不存在,则直接创建后再将其返回
     */
    public static Cache_test valueOf(String data){
        for (int i = 0; i < MAX_SIZE; i++) {
            if (cache[i] != null
                    && cache[i].get_data().equals(data)) {
                return cache[i];
            }
        }
        if(MAX_SIZE == pos){
            cache[0]    = new Cache_test(data);
            pos            = 1;
        }else{
            cache[pos]    = new Cache_test(data);
        }
        return cache[pos++];
    }
}
Cache_test

 

  (2)测试类Main

public class Main {
    
    public static void main(String[] args){
        Cache_test ct1 = new Cache_test("test1");
        Cache_test ct2 = new Cache_test("test1");
        //由于这里都是直接创建,所以下面会输出false;
        System.out.println(ct1 == ct2);
        
        Cache_test ct3 = Cache_test.valueOf("test2");
        Cache_test ct4 = Cache_test.valueOf("test2");
        //由于这里使用的是valueOf()函数,将会使用到缓存。所以下面输出true.
        System.out.println(ct3 == ct4);
    }
}
Main

  上面的例子中,实现原理为:使用一个数组来缓存该类的对象,数组的长度为MAX_SIZE。每一次调用valueOf来创建对象时,缓存池将会先去查找缓存池中是否已经存在该对象,如果存在,则直接返回该对象,所以当输入两个相同data时,返回回来的对象是同一个,所以上面 ct3 和 ct4 为同一个对象。当缓存数组不存在该对象时,缓存池将根据传入的参数创建一个新的对象,再将其存储到缓存数组中。另外,在这里缓存池使用的是“先进先出”的原则。

PS:上面实例中,用于Cache_test的构造函数为共有,所以,允许创建不存储到缓存池中的对象,假如要强制使用缓存池,则可以将构造函数声明为private。

  

  了解了缓存原理后,我们来看看实际JDK中使用了缓存的类。

包装类 Integer 的缓存

类似于我们上面提到的缓存原理,Integer类如果使用new构造函数来创建对象,则每次都将返回全新的对象;假如采用了valueOf方法来创建对象,则会缓存该创建的对象。让我们来看看源码:

private static class IntegerCache {//内部类,注意它的属性都是定义为static final  
    static final inthigh; //缓存上界  
    static final Integer cache[];//cache缓存是一个存放Integer类型的数组  
  
    static {//静态语句块  
        final int low = -128;//缓存下界,值不可变  
  
        // high value may beconfigured by property  
        int h = 127;// h值,可以通过设置jdk的AutoBoxCacheMax参数调整(参见(3))  
        if (integerCacheHighPropValue !=null) {  
            // Use Long.decode here to avoid invoking methods that  
            // require Integer‘s autoboxing cache to be initialized  
            // 通过解码integerCacheHighPropValue,而得到一个候选的上界值  
            int i = Long.decode(integerCacheHighPropValue).intValue();  
            // 取较大的作为上界,但又不能大于Integer的边界MAX_VALUE  
            i = Math.max(i, 127);//上界最小为127  
            // Maximum array size is Integer.MAX_VALUE  
            h = Math.min(i, Integer.MAX_VALUE - -low);  
        }  
        high = h; //上界确定,此时high默认一般是127  
        // 创建缓存块,注意缓存数组大小  
        cache =new Integer[(high - low) + 1];  
        int j = low;  
        for(int k = 0; k <cache.length; k++)  
            cache[k] =new Integer(j++);// -128到high值逐一分配到缓存数组  
    }  
  
    private IntegerCache() {}//构造方法,不需要构造什么
Integer

 

 简单来说,就是使用了一个内部类IntegerCache 来管理缓存cache[]。但使用valueOf()方法时,系统将会判断是否存在于缓存池中。然而,请注意,这里有所不同的是,Integer类在加载时,就已经预先将一部分对象(即从-128到127)创建好了,也就是说每一次调用valueOf方法时,假如传入的值在-127到128之间,则Integer类直接返回已经创建好的对象,假如传入的参数值在此区间之外,则Integer类会创建一个全新的对象。

 

再看小程序

现在,让我们重新回来一开始的小程序。

(1)程序中 i1 和 i2 利用其构造函数进行构造,所以,两者是两个不同的对象,因此返回false。

(2)通过使用javap 查看字节码,可知 i3 和 i4 、i5 和 i6 的自动装箱事实上是调用了valueOf方法。i3 和 i4 的值在-128到127之间,所以直接使用缓存池的对象,而 i5 和 i6 超出该区间,所以创建的是新对象。

由此便可以得知所以输出结果了。

 

后记

通过资料查找和源码的查看,可以知道,除了Integer类外,还有Byte、Short、Long、Character也使用了缓存,而Flot、Double没有使用缓存。

 

相关资料

《Integer中用静态内部类所作的缓存》

《Java中的装箱与拆箱》

《Java 自动装箱和拆箱》