1:== 与 equals()(重要)
== : 它的作用是判断两个对象的地址是不是相等 判断两个对象是不是同一个对象(基本数据类型==比较的是值,引用数据类型==比较的是内存地址)。 equals() : 它的作用也是判断两个对象是否相等。 两种使用情况: 情况 1:类没有覆盖 equals() 方法。则通过 equals() 比较该类的两个对象时,等价于通过“==”比较这两个对象。 情况 2:类覆盖了 equals() 方法。一般,我们都覆盖 equals() 方法判断来两个对象的内容相等; 若它们的内容相等,则返回 true (即,认为这两个对象相等)。 注意:String的equal方法时经过重写的
//String中重写的euqals方法 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; }
2:hashCode()介绍
hashCode()在散列表中才有用,在其它情况下没用。 在散列表中 hashCode() 的作用是获取对象的哈希码(散列码),进而确定该对象在散列表中的位置。 hashCode() 的作用是获取哈希码,也称为散列码; 它实际上是返回一个 int 整数。这个哈希码的作用是确定该对象在哈希表中的索引位置。 hashCode() 定义在 JDK 的 Object.java 中,这就意味着 Java 中的任何类都包含有 hashCode() 函数。 散列表存储的是键值对(key-value),它的特点是:能根据“键”快速的检索出对应的“值”。 以“HashSet 如何检查重复”为例子来说明为什么要有 hashCode: 当你把对象加入 HashSet 时, HashSet 会先计算对象的 hashcode 值来判断对象加入的位置,同时也会与其他已经加入的对象的 hashcode 值作比较; 如果没有相符的 hashcode,HashSet 会假设对象没有重复出现。 如果发现有相同 hashcode 值的对象,这时会调用 equals()方法来检查 hashcode 相等的对象是否真的相同。 如果两者相同,HashSet 就不会让其加入操作成功。 如果不同的话,就会重新散列到其他位置。
3:hashCode()与 equals()
对于hashCode()与 equals()的关系 需要分两种情况讨论第一种 不会创建“类对应的散列表” 这里所说的“不会创建类对应的散列表”是说: 我们不会 不会在HashSet, Hashtable, HashMap等等这些本质是散列表的数据结构中 使用 hashCode() equals() 用来比较该类的两个对象是否相等。 hashCode() 则根本没有任何作用,所以,不用理会hashCode()。
import java.util.*;import java.lang.Comparable;public class NormalHashCodeTest{ public static void main(String[] args) { // 新建2个相同内容的Person对象, // 再用equals比较它们是否相等 Person p1 = new Person("eee", 100); Person p2 = new Person("eee", 100); Person p3 = new Person("aaa", 200); System.out.printf("p1.equals(p2) : %s; p1(%d) p2(%d)\n", p1.equals(p2), p1.hashCode(), p2.hashCode()); System.out.printf("p1.equals(p3) : %s; p1(%d) p3(%d)\n", p1.equals(p3), p1.hashCode(), p3.hashCode()); } private static class Person { int age; String name; public Person(String name, int age) { this.name = name; this.age = age; } /** * @desc 覆盖equals方法 */ public boolean equals(Object obj){ if(obj == null){ return false; } //如果是同一个对象返回true,反之返回false if(this == obj){ return true; } //判断是否类型相同 if(this.getClass() != obj.getClass()){ return false; } Person person = (Person)obj; return name.equals(person.name) && age==person.age; } }}//p1.equals(p2) : true; p1(1169863946) p2(1901116749)//p1.equals(p3) : false; p1(1169863946) p3(2131949076)
第二种 会创建“类对应的散列表” 这里所说的“会创建类对应的散列表”是说: 我们会在HashSet, Hashtable, HashMap等等这些本质是散列表的数据结构中 如果两个对象相等,则 hashcode 一定也是相同的; 两个对象相等,对两个对象分别调用 equals 方法都返回 true; 两个对象有相同的 hashcode 值,它们也不一定是相等的(不同的对象也可能产生相同的 hashcode,哈希冲突); 在这种情况下,除了要覆盖equals()之外,也要覆盖hashCode()函数 hashCode() 的默认行为是对堆上的对象产生独特值。 如果没有重写 hashCode(),则该 class 的两个对象无论如何都不会相等(即使这两个对象指向相同的数据)
public class ConflictHashCodeTest{ public static void main(String[] args) { // 新建Person对象, Person p1 = new Person("eee", 100); Person p2 = new Person("eee", 100); Person p3 = new Person("EEE", 100); // 新建HashSet对象 HashSet set = new HashSet(); set.add(p1); set.add(p2); set.add(p3); // 比较p1 和 p2, 并打印它们的hashCode() System.out.printf("p1.equals(p2) : %s; p1(%d) p2(%d)\n", p1.equals(p2), p1.hashCode(), p2.hashCode()); // 比较p1 和 p3, 并打印它们的hashCode() System.out.printf("p1.equals(p4) : %s; p1(%d) p4(%d)\n", p1.equals(p3), p1.hashCode(), p3.hashCode()); // 打印set System.out.printf("set:%s\n", set); } private static class Person { int age; String name; public Person(String name, int age) { this.name = name; this.age = age; } /** * @desc重写hashCode */ @Override public int hashCode(){ int nameHash = name.toUpperCase().hashCode(); return nameHash ^ age; } /** * @desc 覆盖equals方法 */ @Override public boolean equals(Object obj){ if(obj == null){ return false; } //如果是同一个对象返回true,反之返回false if(this == obj){ return true; } //判断是否类型相同 if(this.getClass() != obj.getClass()){ return false; } Person person = (Person)obj; return name.equals(person.name) && age==person.age; } }}//p1.equals(p2) : true; p1(68545) p2(68545)//p1.equals(p3) : false; p1(68545) p3(68545)// 比较p1和p2,我们发现:它们的hashCode()相等,通过equals()比较它们也返回true。所以,p1和p2被视为相等。//因此p1 set进去后 p2不能set进去//比较p1和p3,我们发现:虽然它们的hashCode()相等;但是,通过equals()比较它们返回false。所以,p1和p3被视为不相等。 //因此p1 set进去后 p3还能set进去
4:== 与 equal 与 hashcode 总结
== :基本数据类型的值 引用数据类型地址equal:没被重写 判断逻辑 与== 一致 被重写判 判断逻辑 为判断内容hashcode :经过hash 运算生成的值1:hashcode不经过重写完全没意义2:如果不是 之后使用在HashSet, Hashtable, HashMap 则hashcode值完全没意义3:在hashcode 经过重写 并且 在之后使用在HashSet, Hashtable, HashMap 如果是则hashcode一致 内容不一定完全相等 对象也就不一定相等(哈希运算算法 字母大写(换小写) 与 字母小写) 在通过重写的equal比较 如果还是一致 对象才一定相等 通常在hash表中会先进行 hashcode的比较(更快) 再进行equal的比较