分析 equals、hashCode 与内存泄露
1、equals
equals的作用:比较两个对象的地址值是否相等
equals() 方法再 object类中定义如下:
public boolean equals(Object obj) {
return (this == obj);
}
equal方法是能重写的。没有重写的情况下 比较的就是地址值
java对equals的要求:
-
对称性:如果 x.equals(y)返回是“true”,那么 y.equals(x)也应该返回是“true”。
-
反射性:x.equals(x)必须返回是“true”。
-
类推性:如果 x.equals(y)返回是“true”,而且 y.equals(z)返回是“true”,那么 z.equals(x)也应该返回是“true”。
-
还有一致性:如果 x.equals(y)返回是“true”,只要 x 和 y 内容一直不变,不管你重复 x.equals(y)多少次,返回都是 “true”。
-
任何情况下,x.equals(null),永远返回是“false”;x.equals(和 x 不同类型的对象)永远返回是“false”。
2、hashcode()
hashcode() 方法,在object类中定义如下:
public native int hashCode();
说明是一个本地方法,它的实现时根据本地机器相关的。
当然我们可以在自己写的类中覆盖hashcode()方法,比如String、Integer、Double。。。这些都是覆盖hashcode方法的。
Java.lang.object 中对 hashCode 的约定:
- 在一个应用程序执行期间,如果一个对象的equals方法做比较所用到的信息没有被修改的话,则对该对象调用hashCode方法多次,它都始终放回同一个整数。
- 如果两个对象根据equals(Object o) 方法是相等的,则调用这两个对象中的任意对象的hashCode 方法必须产生相同的整数结果。
- 如果两个对象根据equals(Object o) 方法不相等, 则调用这两个对象的任意对象的hashCode方法,不要求产生不同的整数结果。但如果可能不同,则可能提高散列表的性能。
在java集合中,判断两个对象是否相等的规则是:
-
判断两个对象的 hashCode 是否相等
如果不相等,认为两个对象也不相等,完毕
如果相等,转入 2(这一点只是为了提高存储效率而要求的,其实理论上没有也可以,但如果没有,实际使用时效率会大大降低,所以我们 这里将其做为必需的。后面会重点讲到这个问题。)
-
判断两个对象用equals运算是否相等如果不相等,
认为两个对象也不相等
如果相等,认为两个对象相等(equals()是判断两个对象是否相等的关键)
注意:
当一个对象被存进 HashSet 集合后,就不能修改这个对象中的那些参与计算的哈希值的字段了,否则,对象被修改后的哈 希值与最初存储进 HashSet 集合中时的哈希值就不同了,在这种情况下,即使在 contains 方法使用该对象的当前引用作为 的参数去 HashSet 集合中检索对象,也将返回找不到对象的结果,这也会导致无法从 HashSet 集合中删除当前对象,从而 造成内存泄露。