有以下一段代码:
Integer a = 123;
Integer b = 123;
int c = 123;
int d = 123;
System.out.println(c == d);
System.out.println(a == b);
System.out.println(a == c);
这段代码运行的结果是什么呢?
c == d 一定为True。
由于Java中存在自动拆装箱,故Integer包装类型和int基本类型是等价的,所以a == c 以及 c == d 也为True。
那如果把值改为1230呢?
Integer a = 1230;
Integer b = 1230;
int c = 1230;
int d = 1230;
System.out.println(a == b);
System.out.println(c == d);
System.out.println(a == c);
哎,此时运行结果变为了 False, True, True。
那为什么 a != b 了呢?
Integer a = 1230;
在 Java 运行中,这行代码在编译完成后会转化为
Integer a1 = Integer.valueOf(1230);
而在 Integer 的 valueOf() 方当中,如果数值在-128-127之间,就都存储在一个数组当中,该数组相当于一个缓存,当我们在-128-127之间进行自动装箱的时候,我们就直接返回该值在内存当中的地址,所以在-128-127之间的数值用==进行比较是相等的。而不在这个区间的数,需要新开辟一个内存空间进行装箱,所以用==比较是不相等的。
这就称为Java128陷阱。
我们看一下 valueOf() 源码:
valueOf() 方法会首先判断是否大于 low 小于 high,如果在这个区间会直接返回 cache 数组。
所以在Integer a = 123;Integer b = 123;中,a 和 b 返回的是同一个对象,所以 == 为 True。
当值为 1230 时,就进入 new Integer ,两个新的 Integer 对象,所以 == 为 False。