决定每天上午8点整,分享一道 Java 面试题。厚积薄发,每天进步一点点!
【第 2 天】题目
== 和 equals 的区别是什么?
参考答案
== 解读:
对于基本类型和引用类型 == 的作用效果是不同的,如下所示:
代码示例:
public static void test1 () {
Integer x = 11;
Integer y = 11;
Integer z = new Integer("11");
Integer z1 = new Integer("11");
System.out.printf("x == y : %s%n", x == y);
System.out.printf("x == z : %s%n", x == z);
System.out.printf("x.equals(y) : %s%n", x.equals(y));
System.out.printf("x.equals(z) : %s%n", x.equals(z));
System.out.printf("z.equals(z1) : %s%n", z.equals(z1));
System.out.printf("z == z1 : %s%n", z == z1 );
}
运行结果:
x == y : true x == z : false x.equals(y) : true x.equals(z) : true z.equals(z1) : true z == z1 : false
代码分析:因为 x 和 y 指向的是同一个引用,所以 == 也是 true,而 new Integer( )方法则重新开辟了内存空间,所以 == 结果为 false,而 equals 比较的一直是值,所以结果都为 true【所以对于包装类型的比较建议用 equals】。
equals 解读:
equals 本质上就是 ==,只不过 Integer 和 String 等重写了 equals 方法,把它变成了值比较。看下面的示例就明白了。
首先来看默认情况下 equals 比较一个有相同值的对象,代码如下:
class TestEquals {
private String str;
public TestEquals (String str) {
this.str = str;
}
}
public void test3 () {
TestEquals t1 = new TestEquals("1");
TestEquals t2 = new TestEquals("1");
System.out.printf("t1.equals(t2) : %s%n", t1.equals(t2));
}
运行结果:
t1.equals(t2) : false
输出结果出乎我们的意料,竟然是 false?这是怎么回事,看了 equals 源码就知道了,源码如下:
public boolean equals(Object obj) {
return (this == obj);
}
原来 equals 本质上就是 ==
那问题来了,两个相同值的 Integer对象,为什么返回的是 true?代码如下:
public static void test4 () {
Integer i1 = new Integer("11");
Integer i2 = new Integer("11");
System.out.printf("i1.equals(i2) : %s%n", i1.equals(i2));
}
运行结果:
i1.equals(i2) : true
同样的,当我们进入 Integer的 equals 方法,找到了答案,代码如下:
public boolean equals(Object obj) {
if (obj instanceof Integer) {
return value == ((Integer)obj).intValue();
}
return false;
}
原来是 Integer 重写了 Object 的 equals 方法,把引用比较改成了值比较。
总结 :== 对于基本类型来说是值比较,对于引用类型来说是比较的是引用;而 equals 默认情况下是引用比较,只是很多类重写了 equals 方法,比如 String、Integer 等把它变成了值比较,所以一般情况下 equals 比较的是值是否相等。