两个对象用 ==
比较,比较的是两个对象的内存地址,而不是他们值,a1
和 a2
粗看就是两个不同的对象,那么结果就是 false,事实如何呢?我们写个程序运行下:
Integer a1 = 100;
Integer a2 = 100;
System.out.println("a1 == a2:" + (a1 == a2));
// 结果...
a1 == a2:true
为什么会是 true 呢?不是两个不同的对象么?这其实是跟 Integer 的实现机制有关。为了提供性能和减少内存的消耗,Java 引入了一个 Integer Cache
机制 ,该机制用于缓存 -128 ~ 127
的 Integer
对象。Integer
中有一个 IntegerCache
的内部类,它就是负责存储这个范围内的 Integer
实例的缓存对象。当我们使用 Integer.valueOf(int)
方法创建 Integer
,它首先会检查该值是否在缓存范围内,如果是,则返回缓存中的对象,如果不是,则创建一个新的 Integer
对象。Integer
的自动装箱是通过 Integer.valueOf(int)
方法实现的。源码如下:
public static Integer valueOf(int i) {
if (i >= IntegerCache.low && i <= IntegerCache.high)
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}
所以上面两个自动装箱产生的 a1 和 a2 其实是同一个对象。如果 a1 = 128
、a2 = 128
呢?结果应该为 false。
扩展
在 Java 中,创建 Integer
对象有五种方式:
- 1、直接 new 一个 Integer 对象。这种方式每次都会创建一个新的
Integer
对象,尽管它已经存在于缓存中了。目前该方式在 Java 9 中标注为已废弃了:
- 2、自动装箱。当将一个
int
赋值给一个Integer
变量时,Java 会自动将其装箱成Integer
对象。 - 3、使用 Integer.valueOf() 方法。这是 Java 推荐创建
Integer
对象的方式。 - 4、使用 Integer.parseInt() 方法。
该方法首先解析字符串以得到基本的 int
值,然后可以通过自动装箱将其转换为 Integer
对象。
Integer num4 = Integer.parseInt("123");
- 5、使用 Integer.decode() 方法
Integer
提供了 decode()
用于将字符串解析为 Integer
,当接受的参数表示整数,该方法返回相应的 Integer
对象。
Integer num5 = Integer.decode("123");