==简介:
首先对于基本类型和引用类型 == 的作用效果是不同的,
基本数据类型(也称原始数据类型) :byte,short,char,int,long,float,double,boolean。
==用于比较引用和比较基本数据类型时具有不同的功能:
-
比较基本数据类型,如果两个值相同,则结果为true
-
而在比较引用时,如果引用指向内存中的同一对象,结果为true
简单概括就是:
基本类型:比较的是值是否相同;
引用类型:比较的是引用是否相同;
代码示例:
/**
* Created by MisterXie on 2020/2/22
* 个人微信公众号:程序员Hotel
*/
public class test {
public static void main(String[] args) {
String x = "str";
String y = "str";
String z = new String("str");
System.out.println(x==y); // true
System.out.println(x==z); // false
System.out.println(x.equals(y)); // true
System.out.println(x.equals(z)); // true
}
}
代码解读:
x==y这个时候,利用了Java 常量池的机制,x和y同时指向的是常量池中的同一块地址区,即 x 和 y 指向的是同一个引用,所以x==y的结果是 true。
而当z=new String()方法则重写开辟了内存空间,所以x==z结果为 false。
当使用equals 的时候,其实比较的一直是值,所以结果都为 true。
equals 解读:equals 本质上就是 ==,只不过 String 和 Integer 等重写了 equals 方法,把它变成了值比较。看下面的代码就明白了。
首先来看默认情况下 equals 比较一个有相同值的对象,代码如下:
/**
* Created by MisterXie on 2020/2/22
* 个人微信公众号:程序员Hotel
*/
public class Hotel {
public Hotel(String name) {
this.name = name;
}
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public static void main(String[] args) {
Hotel c1 = new Hotel("程序员Hotel");
Hotel c2 = new Hotel("程序员Hotel");
System.out.println(c1.equals(c2)); // false
}
//延伸:如果想要结果为true,则需要重写equals和hashCode方法
/* @Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Hotel hotel = (Hotel) o;
return Objects.equals(name, hotel.name);
}
@Override
public int hashCode() {
return Objects.hash(name);
}*/
}
输出结果是 false?这是怎么回事,然后我们可以查看Object的equals 源码,源码如下:
public boolean equals(Object obj) {
return (this == obj);
}
原来 equals 本质上就是 ==。此处定义的equals与==是等效的。
那问题来了,两个相同值的 String 对象,为什么返回的是 true?代码如下:
/**
* Created by MisterXie on 2020/2/22
* 个人微信公众号:程序员Hotel
*/
public class test {
public static void main(String[] args) {
String s1 = new String("程序员Hotel");
String s2 = new String("程序员Hotel");
System.out.println(s1.equals(s2)); // true
}
}
同样的,当我们进入 String 的 equals 方法,找到了答案,代码如下:
public boolean equals(Object anObject) {
if (this == anObject) {
return true;
}
if (anObject instanceof String) {
String anotherString = (String)anObject;
int n = value.length;
if (n == anotherString.value.length) {
char v1[] = value;
char v2[] = anotherString.value;
int i = 0;
while (n-- != 0) {
if (v1[i] != v2[i])
return false;
i++;
}
return true;
}
}
return false;
}
原因就是 String 重写了 Object 的 equals 方法,把引用比较改成了值比较。
总结 :== 对于基本类型来说是值比较,对于引用类型来说是比较的是引用;而 equals 默认情况下是引用比较,只是很多类重写了 equals 方法,比如 String、Integer 等把它变成了值比较,所以一般情况下 equals 比较的是值是否相等。
面试延伸:
问:两个对象的 hashCode() 相同,则 equals() 也一定为 true,对吗?
答:不对,两个对象的 hashCode() 相同,equals() 不一定 true。
问:重写 equals 时要同时重写 hashcode 吗,为什么?
答: 是的,按照 Java 规范,两个对象是 equal 的,那么这两个对象的 hashCode 值也要相等的。
集合框架中,先根据 hashCode 来判断对象是否不相等,只要是不相等,就认为他们肯定不 是同一个对象,不再调用 equals 判断。
因此,要确保 equal 的对象拥有相同的 hashcode 是非常有必要的(反则未必)。