Integer == Integer?

/**
 *Q:Integer和Integer对象用  == 进行比较的过程是怎样的?
 *
 *看例子
 */
public class Test {

	public static void main(String[] args) {
		int a = 1;
		int b = 1;
		Integer c = 3;
		Integer d = 3;
		Integer e = 321;
		Integer f = 321;

		System.out.println(a == b);
		System.out.println(c == d);
		System.out.println(e == f);

	}
}
output:
true
true
false

下面具体解释三个结果
通过java命令   javap -c 得到下面的字节码

 Compiled from "Test.java"
public class Test extends java.lang.Object{
public Test();
  Code:
   0:   aload_0
   1:   invokespecial   #1; //Method java/lang/Object."<init>":()V
   4:   return

public static void main(java.lang.String[]);
  Code:
   0:   iconst_1  //将int类型常量1压入栈 
   1:   istore_1  //将int类型值存入局部变量1
   2:   iconst_1  //将int类型常量1压入栈
   3:   istore_2  //将int类型值存入局部变量2
   4:   iconst_3  //将int类型常量3压入栈
   5:   invokestatic    #2; //Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer; //调用Integer的静态方法 valueOf,构建整型值为3的Integer对象
   8:   astore_3  //将引用存入局部变量3
   9:   iconst_3
   10:  invokestatic    #2; //Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
   13:  astore  4
   15:  sipush  321  //将321压入栈
   18:  invokestatic    #2; //Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
   21:  astore  5
   23:  sipush  321
   26:  invokestatic    #2; //Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
   29:  astore  6
   31:  getstatic       #3; //Field java/lang/System.out:Ljava/io/PrintStream;
   34:  iload_1   //从局部变量1中装载int类型值到栈
   35:  iload_2
   36:  if_icmpne       43  //从栈中pop出两个int类型值并进行大小比较
   39:  iconst_1
   40:  goto    44
   43:  iconst_0
   44:  invokevirtual   #4; //Method java/io/PrintStream.println:(Z)V
   47:  getstatic       #3; //Field java/lang/System.out:Ljava/io/PrintStream;
   50:  aload_3   //从局部变量3中装载引用到栈
   51:  aload   4
   53:  if_acmpne       60   //从栈中pop出两个引用值进行比较
   56:  iconst_1
   57:  goto    61
   60:  iconst_0
   61:  invokevirtual   #4; //Method java/io/PrintStream.println:(Z)V
   64:  getstatic       #3; //Field java/lang/System.out:Ljava/io/PrintStream;
   67:  aload   5
   69:  aload   6
   71:  if_acmpne       78
   74:  iconst_1
   75:  goto    79
   78:  iconst_0
   79:  invokevirtual   #4; //Method java/io/PrintStream.println:(Z)V
   82:  return

}


整型值的比较很容易理解,就是值的大小比较

但是为什么下面的语句会是不同的结果:

		System.out.println(c == d);
		System.out.println(e == f);

从字节码上看都是一样的指令,没有不同的地方
原因就在Integer的方法 valueOf

我们来看这个方法的源码:字节码里调用的就是这个方法: 

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

private static class IntegerCache {
        static final int high;
        static final Integer cache[];

        static {
            final int low = -128;

            // high value may be configured by property
            int h = 127;
            if (integerCacheHighPropValue != null) {
                // Use Long.decode here to avoid invoking methods that
                // require Integer's autoboxing cache to be initialized
                int i = Long.decode(integerCacheHighPropValue).intValue();
                i = Math.max(i, 127);
                // Maximum array size is Integer.MAX_VALUE
                h = Math.min(i, Integer.MAX_VALUE - -low);
            }
            high = h;

            cache = new Integer[(high - low) + 1];
            int j = low;
            for(int k = 0; k < cache.length; k++)
                cache[k] = new Integer(j++);
        }

        private IntegerCache() {}
    }

也就是说jdk 虚拟机里,Integer里弄了一个缓存,对于在 -128—127 之间的数值,会直接使用该缓存里的对象,而不会另new 新对象
 也就是说 Integer c = 3 或者 Integer c = Integer.valueOf(3) ,最终 c 得到的是Integer里的缓存对象
 同理,d也是获得该相同对象因此 进行 c == d 比较时,c和d引用的是同一个对象,因此就true
 而对于321,已经超出缓存范围了,因此 valueOf 方法会生成一个新的Integer对象因此e和f就引用不同 的对象了,进行==比较,当然就false了
 另外,对Integer的缓存,我们在日常开发时,对于小的整型值应该充分利用Integer的缓存对象省去过多的对象创建,回收的操作,这样会极大的提高程序性能