==
(1)如果比较的是基础数据类型,那么 == 比较的是他们的值是否相同
(2)如果比较的是引用数据类型,那么 == 比较的是他们的引用地址是否相同,也就是比较的栈中局部变量表里的 reference 引用是否相同
equals
(1)equals 是一个 Object 类中的方法,从源码可以知道,他比较的是两个对象的引用地址是否相同
(2)但是其他类会重写 Object 类中的 equals 方法,重新定义比较的规则,像 String 中就重写了 equals 方法,他比较的是两个 String 对象的值内容是否相同,也就是比较两个 String 对象指向的堆上实例数据是否相同
总结
(1)当数据是基本数据类型时,只能用 == 来比较他们的值是否相同,不能用 equals 来比较
(2)当 equals 没有发生重写时,equals 和 == 相同,都是比较两个对象的引用地址是否相同
(3)如果 equals 被其他类重写,那么就需要按照重写的规则来比较,例如 String 类将 equals 方法修改为比较两个对象的值内容是否相同
String 当中的 equals 方法源码
(1)首先通过 == 判断两者的地址是否相同,如果相同则为字符串本身,直接返回 true
(2)接着判断 object 是不是 String 类型,不是的话返回 false,是的话再进一步判断
(3)获取两者的长度进行比较,长度不同直接返回 false,相同再继续比较
(4)获取两者的字符数组,通过循环依次判断每一位字符是否相同
/** The value is used for character storage.*/
private final char value[];
public boolean equals(Object anObject) {
//首先判断比较的字符串是不是本身
if(this == anObject) {
return true;
}
//判断Object是否是String类型
if(anObject instanceof String) {
// true 进行强制类型声明
String anotherString = (String) anObject;
//获取char类型数组的长度
int n = value.length;
//比较 anotherString转换成字符数组的长度是否相等
if(n == anotherString.value.length) {
//分布定义两个char[]
char v1[] = value;
char v2[] = anotherString.value;
/*
*循环递减直到字符数组长度直到不为0
* 依次从下标0开始比较两个char[]
*/
int i = 0;
while (n-- != 0) {
if (v1[i] != v2[i])
return false;
i++;
}
return true;
}
}
return false;
}