Integer 值比较 equal 与 ==
Integer 值比较 equal 与 ==
引言
这两天写 CRUD 的逻辑时,碰到个问题。枚举值里的 Integer 和另一个 Integer ,使用==
进行值比较时,明明值一样,返回的却一直是 false,使用equals()
返回才是 true。
之前用==
、equal()
时,返回结果在预期内,就没有留意。虽然知道前者肯定变成比较地址了,但是这方面的知识我想不起来了,翻了翻了基础知识,加深下印象。知道的就没必要读下去了。Integer 之间的比较用 equal ,最好别用 ==。
一些比较
public static void main(String[] args) {
int test = 3;
Integer intA = new Integer(3);
Integer intB = new Integer(3);
Integer intC = 2 + 1;
Integer intD = 52 -49;
Integer intE = 3;
Integer intF = test;
Integer intG = 128;
Integer intH = 128;
System.out.println(intA == intB); // false
System.out.println(intA == intC); // false
System.out.println(intC == intB); // false
System.out.println(intG == intH); // false
System.out.println(intC == intD); // true
System.out.println(intC == intE); // true
System.out.println(intD == intE); // true
System.out.println(intD == intF); // true
}
Integer 内部有一个静态变量池 IntegerCache,存放了一个数组Integer[],-128 ~ 127。普通的两个值运算都会进行自动拆箱,再进行自动装箱,自动装箱使用的是 valueOf 方法。Integer.valueOf 方法在实现时如果不超过范围(-128 ~ 127)则从 IntegerCache 中获取缓存的Integer对象。 如果超过范围则是 new 一个新Integer对象。
所以 new Integer,比较,false;赋值不在(-128 ~ 127)区间,比较,false。
所以,Integer 之间的比较用 equal ,最好别用 ==。
一些源码
Integer
Integer 的自动拆装箱:
Interge i=100;
// 实际代码为
Interge i=Integer.valueOf(100);
而 Integer 的 valueof():
public static Integer valueOf(int i) {
if (i >= IntegerCache.low && i <= IntegerCache.high)
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}
IntegerCahce
此处可以看见,在一定范围内,返回值为 IntegerCahce 内的缓存,最后以下 Integer 的内部类 IntegerCahce,如下:
private static class IntegerCache {
static final int low = -128;
static final int high;
static final Integer cache[];
static {
// high value may be configured by property
int h = 127;
String integerCacheHighPropValue =
sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
if (integerCacheHighPropValue != null) {
try {
int i = parseInt(integerCacheHighPropValue);
i = Math.max(i, 127);
// Maximum array size is Integer.MAX_VALUE
h = Math.min(i, Integer.MAX_VALUE - (-low) -1);
} catch( NumberFormatException nfe) {
// If the property cannot be parsed into an int, ignore it.
}
}
high = h;
cache = new Integer[(high - low) + 1];
int j = low;
for(int k = 0; k < cache.length; k++)
cache[k] = new Integer(j++);
// range [-128, 127] must be interned (JLS7 5.1.7)
assert IntegerCache.high >= 127;
}
private IntegerCache() {}
}
代码中可见,Integer把-128到127(可通过JVM参数配置进行调整,增大范围)的整数都提前实例化了。所以不管创建多少个这个范围内的 Integer 用 valueOf 出来的都是同一个对象。此举是为了尽可能的减少 new() 带来的开销。
在 Long 和 Short 的源码中,也做了同样的优化。