java.lang.String类的equals和hashCode方法
程序员文章站
2022-03-15 20:34:17
...
String 类代表字符串,也是 Java 中一个封装好的类。Java 程序中的所有字符串字面值(如 “abc” )都作为此类的实例实现。
String 类的底层封装了一个 final 的字符数组。基于这个数组,说明字符串是常量,它们的值在创建之后不能更改。
String 的方法基本都是基于对这个字符数组来进行处理和实现的。
属性
// 字符数组
private final char value[];
// 哈希值
private int hash;
equals方法重写
String 的 equals 方法进行了重写,只要两个字符串各个位置的字符完全一样,且字符数组长度一样,就相等。
public boolean equals(Object anObject) {
if (this == anObject) {
// 如果当前字符串对象和目标对象引用的是内存中同一位置,返回true
return true;
}
if (anObject instanceof String) {
String anotherString = (String)anObject;
// 当前字符串的长度n
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;
}
hashCode方法重写
hash 是私有属性,初始为 0,当第一次调用此方法时,进行计算。以后再次调用,就不再重新计算。
由循环可以推导出 jdk 中注释的 s[0]*31^(n-1) + s[1]*31^(n-2) + … + s[n-1] 这个公式。
采用 31 这个数字,可以参考这个问题:https://*.com/questions/299304/why-does-javas-hashcode-in-string-use-31-as-a-multiplier
和另一篇深入分析的文章:科普:为什么 String hashCode 方法选择数字31作为乘子
public int hashCode() {
int h = hash;
if (h == 0 && value.length > 0) {
char val[] = value;
// 遍历字符数组中每一个字符
for (int i = 0; i < value.length; i++) {
h = 31 * h + val[i];
}
hash = h;
}
return h;
}