先来看一段代码
public class IntegerDemo01 { public static void main(String[] args) { Integer i2 =100; Integer i3 =100; System.out.println(i2==i3); Integer i4 =1000; Integer i5 =1000; System.out.println(i3==i4); } }以上输出是true 还是false呢? 其中第一个是true ,第二个是false。初学者就会对这个答案产生疑问。为啥会出现这个情况呢? 先通过反编译 查看一下到底编译器是怎么处理这行代码的?
import java.io.PrintStream; public class IntegerDemo01 { public IntegerDemo01() { } public static void main(String args[]) { Integer i2 = Integer.valueOf(100); Integer i3 = Integer.valueOf(100); System.out.println(i2 == i3); Integer i4 = Integer.valueOf(1000); Integer i5 = Integer.valueOf(1000); System.out.println(i3 == i4); } }从上面代码可以得知,编译器引用变量的定义和初始化中使用到了Integer.valueOf()方法,这个valueOf()是干嘛用的呢?
public static Integer valueOf(int i) { if (i >= IntegerCache.low && i <= IntegerCache.high) //如果数值在-128~127之间的 return IntegerCache.cache[i + (-IntegerCache.low)]; //返回 return new Integer(i); }来看这段代码的描述 * Returns an {@code Integer} instance representing the specified {@code int} value., 返回了这个Integer的含有指定的值的实例 * as this method is likely to yield significantly better space and time performance by * 显着更好的空间和时间性能 * caching frequently requested values.
有人会联想到String类型,因为String的实例有时候也是这么赋值的。那么编译器在背后有没有进行转化呢?看一下代码?
String str = "abc"; String str2 ="abc"; System.out.println(str==str2); String str3 = new String("abc"); String str4 = new String("abc"); System.out.print(str3==str4);如上的结果第一个是true ,第二个是false。第一个是在常量池中去查找是否有没有这个abc这个值,有的话,直接进行引用。没有的话,再在常量池中创建。所以第一个的引用变量他们的地址都相同的,因此为true。 第二个用到了new,这个单词告诉了编译器 直接在堆内存中给我开辟了一个空间,不管之前是否有存在。因此,尽管值也一样,但是它们引用的地址是不一样的。因此为false。 再来看一下编译器是否有将String的实例构建进行某些转换?
String str = "abc"; String str2 = "abc"; System.out.println(str == str2); String str3 = new String("abc"); String str4 = new String("abc"); System.out.print(str3 == str4);发现没有任何操作,没有偷着我们干了什么事。
建议,如果要判断包装类的对象是否相等,最好用equals()方法.
以上大体想法来源于知乎专栏 -java那些事