本文就java语言来讲,探讨一下项目中遇到的坑,以及如何避免这个问题。
首先说一下我遇到的坑,我用Integer进行比较,且比较符号为 =,然后我就debug,找半天最后发现这个问题。
那为什么Integer不能用等于号呢?首先先得说一下装箱和拆箱。
装箱
int 这种基本类型转成Integer对象,可以通过如下代码转换
// 显式转换
int a=7;
Integer aNum = Integer.valueOf(a);
// 隐式转换
Integer b=7;
拆箱
Integer对象转换成int
Integer c=new Integer(6);
int d=c.intValue();
讲完装拆箱,再来说一下new 关键字,new Integer,会直接在堆内存中分配一对象
Integer a=new Integer(5);
Integer b=new Integer(5);
a,b是两个对象,他们在内存中的地址是不一样的。
== 和equals的区别
如果比较的是基本类型的情况下,==,就是比较的内容,而如果是Integer这种对象的情况下,比较的是引用,也就是内存地址,而equal是Object才有的方法,一般都需要手动实现,他比较的是对象的内容
public boolean equals(Object obj) {
if (obj instanceof Integer) {
return value == ((Integer)obj).intValue();
}
return false;
}
通过上面也说明了一个问题,如果两个Integer进行比较使用==比较的是两个地址,所以会返回false,所以我们需要使用equals,但是<,<=,>,>= 是不受影响的。
那Integer和int比较会出现什么问题呢?首先看几个例子。
Integer a=5;
int b=5;
Integer c=128;
int d=128;
Integer e=128;
Integer f=5;
a==b? true,因为Integer和int比较会转换成int类型
c==d? true,同样会转成int类型进行比较
a==f? true
c==e? false 为啥a==f,c!=e呢?看一下源码
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() {}
}
public static Integer valueOf(int i) {
if (i >= IntegerCache.low && i <= IntegerCache.high)
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}
通过源码可以知道,为了更好的空间和时间性能,Integer会缓存频繁使用的数值,数值范围为-128到127,在此范围内直接返回缓存值。IntegerCache.low 是-128,IntegerCache.high是127,如果在这个区间内,他就会把变量i当做一个变量,放到内存中;但如果不在这个范围内,就会去new一个Integer对象,
而如果两个Integer值都不在这个范围内,那么就会new了两个对象实例,两个对象用==比较肯定是false。
String a=new String("abc") ;
String b=new String("abc") ;
String c="abc";
String d="abc";
System.out.println(a==b); false
System.out.println(c==d); true
JVM 会把内存存在两个区域,堆和字符串池,一般new 的对象都会放到堆中,而String c="abc"会把c这个对象放到字符串池中。
String t=new String(“abc”),会涉及到几个动作。 采用new关键字新建一个字符串对象时,JVM首先在字符串池中查找有没有"abc"这个字符串对象,如果有,则不在池中再去创建"abc"这个对象了,直接在堆中创建一个"abc"字符串对象,然后将堆中的这个"abc"对象的地址返回赋给引用,这样,a就指向了堆中创建的这个"abc"字符串对象;如果没有,则首先在字符串池中创建一个"abc"字符串对象,然后再在堆中创建一个"abc"字符串对象,然后将堆中这个"abc"字符串对象的地址返回引用,这样,t指向了堆中创建的这个"abc"字符串对象。
采用字面值的方式创建一个字符串时,JVM首先会去字符串池中查找是否存在"abc"这个对象,如果不存在,则在字符串池中创建"abc"这个对象,然后将池中"abc"这个对象的引用地址返回给字符串常量c,这样c会指向池中"abc"这个字符串对象;如果存在,则不创建任何对象,直接将池中"abc"这个对象的地址返回,赋给字符串常量。
一下是Integer和int对比,如果感觉有用,可以点个关注,以后输出更多有用的内容。