public class Demo {
public static void main(String[] args) {
Integer a1 =128;
Integer a2 =128;
System.out.println((a1==a2)); //false
String I3 ="100";
String I4 = "1"+new String("00");
String I5 ="1"+"00";
System.out.println(I3==I4); //false
System.out.println(I3==I5); //true
Integer I6= 100;
Integer I7 =100;
System.out.println(I6 == I7); //true
}
}
当我们给一个Integer赋予一个int类型的时候会调用Integer的静态方法valueOf。
Integer a1 = Integer.valueOf(128);
Integer a2 = Integer.valueOf(128);
Integer I6 = Integer.valueOf(100);
Integer I7 = Integer.valueOf(100);
那么Integer.valueOf()返回的Integer是不是是重新new Integer(num);来创建的呢?如果是这样的话,那么== 比较返回都是false,因为他们引用的堆地址不一样。给Interger 赋予的int数值在-128 - 127的时候,直接从*cache获取,这些cache引用对Integer对象地址是不变的,但是不在这个范围内的数字,则new Integer(i) 这个地址是新的地址,不可能一样的.
[转载至]
(https://blog.csdn.net/q3838418/article/details/77577490)
若题中代码改为 Integer i6 = new Integer(100);Integer i7 = new Integer(100); 则i6i7 结果为false,因为两个对象之间比较并不会自动拆箱,所以比的是对象的引用地址,故不相等。而原来题目中相等是因为当自动装箱时,JVM会优化,出于节省内存的考虑,JVM会缓存-128到127的Integer对象,所以当自动装箱的数值在这区间时,所装箱的结果都是对缓存池中同一个对象的引用,故==比较地址会为true
关于String类型:String str1=”java”; //指向字符串池
String str2=”blog”; //指向字符串池
String s = “a”+“bc”:编译阶段会直接将“a”和“bc”结合成“abc”,这时如果方法区已存在“abc”,则将s的引用指向该字符串,如不存在,则在方法区中生成字符串“abc”对象,然后再将s的引用指向该字符串
Jvm确实对型如String str1=”java”;的String对象放在常量池里,但是它是在编译时那么做的,而String s=str1+str2;是在运行时刻才能知道,也就是说str1+str2是在堆里创建的,所以结果为false了
拓展与总结:
在String中 == 是判断地址是否相等
关于equals这个方法,在String类中是被重写了的,可以先判断地址是否相同,再比较值是否相同
并不是所有类的equals都是比较值,例如object类的equals方法和==是一样的 注:instanceof是用来判断前后两个对象是否是一个家族中的,也就是判断是否有继承关系
在Integer中,使用 == 来作比较两个对象时(和常数进行比较时,是直接比较值是否相同),需要注意的一点是:对Integer对象进行初始化赋值时,是通过调用valueOf来实现的。
而对于-128到127之间的数(最小值-128是确定了的,但是最大值127是可以通过虚拟机的配置文件来配置),Java会对其进行缓存。而超出这个范围则新建一个对象。
也就是说,如果第一次创建值时在缓存范围之内,第二次如果也是第一次相同的数,其实就是直接在缓存中取的,并没有新建对象,地址相同。