项目场景:
在平常敲代码的时候可能会单纯的认为基本类型可以用 == 进行值的比较,这句话本身没有什么问题,但在有装拆箱环节中我们需要格外注意比较方式,因为装箱就意味着被装箱的变量将不再是基本类型,在这个基础上就不能使用 == 进行判断了,但这也并不绝对,具体还是要看实际环境
问题描述:
在Integer或者Long变量中,如果要使用 == 进行值的比较,则需要注意数值是否在-128~127这个范围内,如果不在则不能比较,以下源码可以解释疑问:
首先我们看先int装箱的代码:
从代码中可以明确的看出并不是所有传入的int变量都进行统一处理,而是在一个范围内的进行统一处理(可从IntegerCache源码中得知这个范围是-128~127),范围外的进行统一处理,范围外的使用的是new 关键字创建的对象,如果是new 出来的则就代表它存在与堆中并且不为基本数据类型了,故不能再使用 == 进行值的比较了
public static Integer valueOf(int i) {
if (i >= IntegerCache.low && i <= IntegerCache.high)
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}
进入IntegerCache看一下源码,Integer.MAX_VALUE的值为2 ^31 -1,从中可以看出,在static静态代码块中创建了一个[-128,127]的缓存池,这个缓存池的存在可以省去我们构建-128~127 之间的数的对象,可以提高效率,
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() {}
}
总结:
提示:经过查看Long的源代码,发现long的装箱与int的装箱是一样的,都存在-128~127的范围问题,所以在对不在范围的数值进行比较时要使用equals