== 和 equals 这两个的意思就是等于,只是equals使用了英语状态,而且对于基本类型和引用类型 == 的作用效果是不同的。
- 基本类型:比较的是值是否相同。
- 引用类型:比较的是引用是否相同;
拓展:那么什么是基本类型和引用类型呢?
1、基本数据类型包括:数值型,字符型,布尔型;
2、引用数据类型:类、接口类型、数组类型、枚举类型、注解类型;
3、基本数据类型和引用数据类型的区别主要在存储方式上:基本数据类型在被创建时,在栈上给其划分一块内存,将数值直接存储在栈上,基本类型是具体的值;引用类型指向内存空间的引用,是个对象。它在被床架时,首先要在栈上给其引用(句柄)分配一块内存,而对象的具体信息都存储在堆内存上,然后由栈上面的引用指向堆中对象的地址。
好了了解完了基本类型和引用类型,那么我们看个示例:
String x = "小明";
String y = "小明";
String z = new String("小明");
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 指向的是同一个引用String,所以 x==y 为true,而 new String()方法则重写开辟了内存空间,所以 x == z 为false,而 equals 比较的一直是值,所以结果都为 true。
拓展: String x = “小明” 和 String z = new String(“小明”)一样吗?
不一样,因为内存的分配方式不一样。String x = "小明"的方式,Java虚拟机会将其分配到常量池中;而String z = new String(“小明”)则会被分到堆内存中。
equals 深入了解:
equals 本质上就是 ==,只是String,Integer 等重写了equals 方法,然后equals 就变成了值的比较。
现在我们来看看默认情况下equals比较一个相同值的对象,结果会怎么样呢?
package com.gx.exception;
public class EqualsAndTwoEqualsDemo {
public static void main(String[] args) {
ComPareEquals comPareEquals1 = new ComPareEquals("小王");
ComPareEquals comPareEquals2 = new ComPareEquals("小王");
System.out.println("comPareEquals1的值= "+comPareEquals1);
System.out.println("comPareEquals2的值= "+comPareEquals2);
System.out.println("comPareEquals1和comPareEquals2比较后的结果:"+comPareEquals1.equals(comPareEquals2));
}
}
class ComPareEquals{
public ComPareEquals(String name){
this.name = name;
}
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
结果为什么会是false呢,不是说比较内存里面的值吗?那么我们来看看打印的结果。
comPareEquals1的值= com.gx.exception.ComPareEquals@4ec4d412
comPareEquals2的值= com.gx.exception.ComPareEquals@5c3f1224
comPareEquals1和comPareEquals2比较后的结果:false
可以看到comPareEquals1和comPareEquals2输出来的值不一样,我们再来看看equals源码:
public boolean equals(Object obj) {
return (this == obj);
}
由上面的 this == obj 可以看出 equals 本质上就是 ==。
那为什么两个相同值的 String 对象,返回的是 true?
String str1 = new String("小红");
String str2 = new String("小红");
System.out.println("str1的值= "+str1);
System.out.println("str2的值= "+str2);
System.out.println("str1和str2比较后的结果:"+str1.equals(str2));
// 结果为true
输出:
str1的值= 小红
str2的值= 小红
str1和str2比较后的结果: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 比较的是值是否相等。