Integer判等的陷阱:你知道Integer内部高速缓冲区IntegerCache吗?

https://blog.csdn.net/magician_Code/article/details/51469101

 

我们先来看看下面代码的运行情况:

public static void main(String[] args)
{
    // TODO Auto-generated method stub
    Integer integer1;
    Integer integer2;

    integer1 = new Integer(10);
    integer2 = new Integer(10);
    //第一次比较
    System.out.println("第一次比较:"+(integer1==integer2));

    //第二次比较
    System.out.println("第二次比较:"+(integer1==10));

    integer1 = 127;
    integer2 = 127;
    //第三次比较
    System.out.println("第三次比较:"+(integer1==integer2));

    integer1 = 128;
    integer2 = 128;
    //第四次比较
    System.out.println("第四次比较:"+(integer1==integer2));
}

运行程序,结果如下:

你看出了运行结果了吗?

第一次和第二次比较就无可厚非了,第一次是直接把两个不同的对象比较,当然是false;第二次比较时,是把Integer对象和int型10进行比较,根据自动装箱、拆箱机制,这时候的比较就等价于10==10,所以是true。那么后面两个为什么会出现两种不同的结果呢?

首先我们先来看看Integer的两种定义方式:

Integer integer1 = new Integer(10);
Integer integer2 = 10;

第一种是我们常见的创建一个对象的方法,那么第二个方法呢?根据Java的自动装箱、拆箱机制,这时在Integer内部实际上是做了如下操作:

Integer integer2 = Integer.valueOf(10);

这时我们查看Integer源码中关于valueOf方法的定义:

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

这里我们会留意到”IntegerCache”这个类,跟踪一下代码,发现这是Integer的一个私有内部类,声明如下:

private static class IntegerCache
  {
    static final int low = -128;
    static final int high;
    static final Integer[] cache;

    private IntegerCache() {}

    static
    {
      int i = 127;
      String str = VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
      if (str != null) {
        try
        {
          int j = Integer.parseInt(str);
          j = Math.max(j, 127);
          i = Math.min(j, 2147483518);
        }
        catch (NumberFormatException localNumberFormatException) {}
      }
      high = i;
      cache = new Integer[high - -128 + 1];
      int k = -128;
      for (int m = 0; m < cache.length; m++) {
        cache[m] = new Integer(k++);
      }
      assert (high >= 127);
    }
  }
}

这段代码并不难读,这里类缓存了从-128到127之间的所有整型对象,意思是当使用自动装箱的方式定义一个值在-128到127的Integer对象时,我们得到的是从缓存区(IntegerCache)中返回的实例。

所以,当我们在进行上面的第三次比较时,此时的integer1和integer2是同一个对象,那么比较的结果当然是true啦。第四次是因为我们指定的值为128,>127,所以Integer内部会创建新的对象返回,所以当然不可能相等。

最后啰嗦一下,如果要进行两个Integer对象基于数值的比较时,因为Integer实现了Compaeable接口,所以直接使用compaerTo方法进行比较会比较妥当。判等的话还可以使用equals方法,于是我们把最开始的代码改成如下:

public static void main(String[] args)
{
    // TODO Auto-generated method stub
    Integer integer1;
    Integer integer2;

    integer1 = new Integer(10);
    integer2 = new Integer(10);
    //第一次比较
    //System.out.println("第一次比较:"+(integer1==integer2));
    if(integer1.equals(integer2))
        System.out.println("第一次比较:"+true);
    else
        System.out.println("第一次比较:"+false);

    //第二次比较
    System.out.println("第二次比较:"+(integer1==10));

    integer1 = 127;
    integer2 = 127;
    //第三次比较
    //System.out.println("第三次比较:"+(integer1==integer2));
    if(integer1.equals(integer2))
        System.out.println("第三次比较:"+true);
    else
        System.out.println("第三次比较:"+false);

    integer1 = 128;
    integer2 = 128;
    //第四次比较
    //System.out.println("第四次比较:"+(integer1==integer2));
    if(integer1.equals(integer2))
        System.out.println("第四次比较:"+true);
    else
        System.out.println("第四次比较:"+false);
}

 

posted @ 2019-03-06 12:30  crazyYong  阅读(463)  评论(0编辑  收藏  举报