论字符串比较常量字符串放前的重要性

1,979 阅读2分钟

字符串比较中常用的三种情况

  • 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的时候要先做一步判空操作。以保证安全避坑。 在一些操作规范的公司里 一般会有封装更好更安全的对象 如判空 断言等基础类以供使用。