- equals 比较字符串内容是否相等
- == 比较字符串的地址是否相同(即两个对象是否是同一个)
- compareTo 比较字符串的大小 -->字典序
==和compareTo我们在使用过程中一般不会出现问题。本小菜鸡主要分享一下那些年equals使用中存在的坑点:空指针异常出及源码解释。
例子很简单:public static void main(String[] args) {
String str1 = null;
String str2 = "出bug了嘻嘻";
System.out.println(str2.equals(str1));
System.out.println(str1.equals(str2));
}
运行结果 :
预期结果是false 可是str1和str2在equals前后的顺序导致了异常的出现。分析一下equals的源码:
equals方法大致工作步骤: 先判断是否是自身比较,如果不是 接着判断anObject是否是字符串。 如果是 则判断字符串内容。 读过String源码我们知道String的内容是放在一个final类型的value数组中的。通过逐一对比两个字符串中的char字符是否相等。 那么两个字符串在equals中先后的顺序就显得尤为重要
- 正确姿势:字符串常量str2在前null字符串str1在后, 这时候传入的null为 anObject 在if (anObject instanceof String)这一行 null 不是String类型直接返回false。
- 错误姿势:null字符串str1在前字符串常量str2在后,此时调用null字符串的equals方法 传入的是字符串常量。可以通过前两个if的判断。 在比较字符串内容时出现空指针异常。因为 null不指向仍和指针。此时 v1[]数组为null。v2[]有值。在if (v1[i] != v2[i]) 该行取值时出现空指针错误。
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;
}
总结:我们在使用equals比较两个字符串是否相等的时候。一般要将不为空的字符串常量放在equals前面。同时,在使用equals的时候要先做一步判空操作。以保证安全避坑。 在一些操作规范的公司里 一般会有封装更好更安全的对象 如判空 断言等基础类以供使用。