使用equals进行内容比较的解释说明
equals是用来判断内容是否相等的。它是Object的方法。Object是所有类的直接或间接父类。
直接使用equals来判断两个引用类型是否相等,其实是判断两个引用类型的地址是否一样。
String str=new String("asd");
String str1=new String("asd");
System.out.println(str.equals(str1));//true
我们知道上述代码执行的结果为true。既然String也是引用类型,那么用equals来判断两个字符串的内容是否相等为什么可以呢?
因为在String类中重写了equals方法,重写之后比较的就不再是两者的地址了。我们来看一下源码:
public boolean equals(Object anObject) {
if (this == anObject) {
return true;
}
if (anObject instanceof String) {
String aString = (String)anObject;
if (coder() == aString.coder()) {
return isLatin1() ? StringLatin1.equals(value, aString.value)
: StringUTF16.equals(value, aString.value);
}
}
return false;
}
看不懂没关系,我们只要记住在String中进行字符串内容的比较是依赖于String中对equals的重写就可以了。
在使用equals进行两个对象的比较时,也是比较的两个对象的地址是否一样。
Person person = new Person("张三",18);
Person person2 = new Person("张三",18);
System.out.println(person.equals(person2));//false`
上述代码的结果为false.就是因为上述两个变量比较的是地址。
如果想自定义两个对象的属性完全相同就认为这两个对象相同,就必须重写equals方法。比如 以下对equals的重写。
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Person person = (Person) o;
return age == person.age &&
Objects.equals(name, person.name);
}
这时再用
Person p = new Person("张三",18);
Person p2 = new Person("张三",18);
System.out.println(p.equals(p2));//true
结果就是true了。这个时候就把对象里面的年龄和姓名进行比较,如果都相同,则认为这两个对象相等。
其实这种比较使用工具类Objects更好,因为如果p2=null,p=null。运行时会发生空指针异常。
但是用Objects的equals(Object a, Object b)方法就可以正常运行因为它的equals源码为:
public static boolean equals(Object a, Object b) {
return (a == b) || (a != null && a.equals(b));
}
也就是当两个对象都为空时,认为两个对象的地址是一样的,a==b就成立了,也符合两个对象为空内容相等的认知。
如果两个对象的地址不一致,则会判断a.equals(b)。需要注意的是:此时的出现了多态的知识,Object a=p,Object b=p2.根据编译看左边运行看右边的原则,equals是p重写后的方法,所以是比较的两个对象里面的内容。而如果p没有重写equals方法,则用的是Object里面的equals方法,那么比较的还是两个对象的地址,此时结果还是为false。因此,使用Objects工具类的前提就是重写Object的equals方法。
下一篇: 3.排序题与sort函数的应用
推荐阅读