為什麼equals()方法要重寫?
判斷兩個物件在邏輯上是否相等,如根據類別的成員變數來判斷兩個類別的實例是否相等,而繼承Object中的equals方法只能判斷兩個引用變數是否是同一個物件。這樣我們往往需要重寫equals()方法。
我們在一個沒有重複物件的集合中加入元素時,集合中存放的往往是對象,我們需要先判斷集合中是否存在已知對象,這樣就必須重寫equals方法。
怎麼重寫equals()方法?
重寫equals方法的要求:
1、自反性:對於任何非空參考x,x.equals(x)應該回傳true。
2、對稱性:對於任何引用x和y,如果x.equals(y)回傳true,那麼y.equals(x)也應該回傳true。
3、傳遞性:對於任何引用x、y和z,如果x.equals(y)回傳true,y.equals(z)回傳true,那麼x.equals(z)也應該回傳true 。
4、一致性:如果x和y所引用的物件沒有發生變化,那麼反覆呼叫x.equals(y)應該會回傳同樣的結果。
5、非空性:對於任意非空參考x,x.equals(null)應該回傳false。
1、自反性原則
在JavaBean中,常常會覆寫equals方法,從而根據實際業務情況來判斷兩個物件是否相等,例如我們寫一個person類,根據姓名來判斷兩個person類別實例物件是否相等。程式碼如下:
1 public class Person { 2 private String name; 3 4 public Person(String name) { 5 this.name = name; 6 } 7 8 public String getName() { 9 return name;10 }11 12 public void setName(String name) {13 this.name = name;14 }15 16 @Override17 public boolean equals(Object obj) {18 if (obj instanceof Person) {19 Person person = (Person) obj;20 return name.equalsIgnoreCase(person.getName().trim());21 }22 return false;23 }24 25 public static void main(String[] args) {26 Person p1 = new Person("张三");27 Person p2 = new Person("张三 ");28 List<Person> list = new ArrayList<Person>();29 list.add(p1);30 list.add(p2);31 System.out.println("是否包含张三:" + list.contains(p1));32 System.out.println("是否包含张三:" + list.contains(p2));33 }34 }
list中含有這個生成的person對象,結果應該是true,但是實際結果:這裡考慮了字串空格的問題,去除前後的空格。
是否包含張三:true
是否包含張三:false
#第二個為什麼會是false呢?
原因在於list中檢查是否含有元素時是透過呼叫物件的equals方法來判斷的,也就是說contains(p2)傳遞進去會依序執行p2.equals(p1)、p2.equals(p2 ),只要一個回傳true,結果就是true。但是這裡p2.equals(p2)回傳的是false?由於我們對字元前後進行了空格的切割造成p2.equals(p2)的比較實際上是:「張三 」.equals(「張三」),一個有空格,一個沒有空格就出錯了。
這個違反了equals的自反性原則:對於任何非空引用x,x.equals(x)應該回傳true。
這裡只要去掉trim方法就可以解決。
2、對稱性原則
上面這個例子,還不是很好,如果我們傳入null值,會怎麼樣呢?增加一條語句:Person p2=new Person(null);
一個null值,所以呼叫name.equalsIgnoreCase()方法時就會報空指標異常。
應該在equals方法裡加上是否為null值的判斷:
是否包含张三:trueException in thread "main" java.lang.NullPointerException//空指针异常3、傳遞性原則
現在我們有一個Employee類別繼承自person類別:
1 @Override 2 public boolean equals(Object obj) { 3 if (obj instanceof Person) { 4 Person person= (Person) obj; 5 if (person.getName() == null || name == null) { 6 return false; 7 }else{ 8 return name.equalsIgnoreCase(person.getName()); 9 }10 }11 return false;12 }
只有在name和ID都相同的情況下才是同一個員工,避免同名同姓的。在main定義了,兩個員工和一個社會閒雜人員,雖然同名同姓但肯定不是同一個人。運行結果應該三個都是false才對。但是:
true
false
p1盡然等於e1,也等於e2,不是同一個類別的實例也相等了?
因為p1.equals(e1)是呼叫父類別的equals方法進行判斷的它使用instanceof關鍵字檢查e1是否是person的實例,由於employee和person是繼承關係,結果就是true了。但放過來就不成立,e1,e2就不等於p1,這也是違反對稱性原則的典型案例。
e1竟然不等於e2?
e1.equals(e2)呼叫的是Employee的equals方法,不僅要判斷姓名相同還有判斷工號相同,兩者的工號不同,不相等時對的。但是p1等於e1,也等於e2,e1卻不等於e2,這裡就存在矛盾,等式不傳遞是因為違反了equals的傳遞性原則:對於實例物件x、y、z;如果x.equals(y)回傳true,y.equals(z)回傳true,那麼x.equals(z)也應該回傳true。
上述情況會發生是因為父類別使用instanceof關鍵字(是否是這個特定類別或它的子類別的一個實例),用來判斷是否是一個類別的實例物件的,這很容易讓子類別「鑽空子」。
想要解決也很簡單,使用getClass進行類型的判斷,person類別的equals方法修改如下:
1 public class Employee extends Person{ 2 private int id; 3 4 5 public int getId() { 6 return id; 7 } 8 public void setId(int id) { 9 this.id = id;10 }11 public Employee(String name,int id) {12 super(name);13 this.id = id;14 // TODO Auto-generated constructor stub15 }16 @Override17 public boolean equals(Object obj) {18 if(obj instanceof Employee){19 Employee e = (Employee)obj;20 return super.equals(obj) && e.getId() == id;21 }22 return super.equals(obj);23 }24 25 public static void main(String[] args){26 Employee e1=new Employee("张三",12);27 Employee e2=new Employee("张三",123);28 Person p1 = new Person("张三");29 30 System.out.println(p1.equals(e1));31 System.out.println(p1.equals(e2));32 System.out.println(e1.equals(e2));33 }34 }
4、必须覆写hashCode方法这样结果就是三个false。
覆写equals方法就必须覆写hashCode方法,这是Javaer都知道的。
原因就是HashMap的底层处理机制是以数组的方式保存map条目的,这其中的关键是这个数组下标的处理机制:
依据传入元素的hashCode方法的返回值决定其数组的下标,如果该数组位置上已经有了map条目,且与传入的键值相等则不处理,若不相等则覆盖;如果数组位置没有条目,则插入,并加入到map条目的链表中。同理检查键是否存在也是根据哈希吗确定文职,然后遍历查找键值的。
那么对象的hashCode方法返回的是什么呢?
他是一个对象的哈希码,是有Object类的本地方法生成的,确保每个对象有一个哈希码。
1、重写equals方法实例 部分代码参考
重写equals方法的目的是判断两个对象的内容(内容可以有很多,比如同时比较姓名和年龄,同时相同的才是用一个对象)是否相同。
如果不重写equals,那么比较的将是对象的引用是否指向同一块内存地址,重写之后目的是为了比较两个对象的value值是否相等。特别指出利用equals比较八大包装对象,(如int,float等)和String类(因为该类已重写了equals和hashcode方法)对象时,默认比较的是值,在比较其它自定义对象时都是比较的引用地址。
package com.lk.C;class User {private String name;private int age;public int getAge() {return age; }public void setAge(int age) {this.age = age; }public void setName(String name) { this.name = name; }public String getName() { return name; }public boolean equals(Object obj) { if(this == obj) { return true; } if(null == obj) { return false; } if(this.getClass() != obj.getClass()) { return false; } User user = (User) obj; if(this.name.equals(user.name)&&this.age == user.age) { return true; } return false; } } public class Test6 { public static void main(String[] args) { User userA = new User(); userA.setName("王明"); userA.setAge(10); User userB = new User(); userB.setName("王明"); userB.setAge(10); User userC = new User(); userC.setName("王亮"); userC.setAge(10); System.out.println("userA equals userB:" + userA.equals(userB)); System.out.println("userA equals userC:" + userA.equals(userC)); } }
userA equals userB:trueuserA equals userC:false
在Java中,问什么说重写了equals方法都要进而重写Hashcode方法呢?
原因如下:当equals此方法被重写时,通常有必要重写 hashCode 方法,以维护 hashCode 方法的常规协定,该协定声明相等对象必须具有相等的哈希码。如下:
(1)当obj1.equals(obj2)为true时,obj1.hashCode() == obj2.hashCode()必须为true
(2)当obj1.hashCode() == obj2.hashCode()为false时,obj1.equals(obj2)必须为false
hashcode是用于散列数据的快速存取,如利用HashSet/HashMap/Hashtable类来存储数据时,都是根据存储对象的hashcode值来进行判断是否相同的。
这样如果我们对一个对象重写了euqals,意思是只要对象的成员变量值都相等那么euqals就等于true,但不重写hashcode,那么我们再new一个新的对象,当原对象.equals(新对象)等于true时,两者的hashcode却是不一样的,由此将产生了理解的不一致。
2、看看下面的三段程序
package com.lk.C;public class Test7 {public static void main(String[] args) {int a = 10;int b = 10; System.out.print("基本类型a==b:"); System.out.println(a == b); System.out.println("-----"); String s1 = "abc"; String s2 = "abc"; System.out.print("String类型是s1==s2:"); System.out.println(s1 == s2); System.out.println("-----"); String s3 = new String("abc"); String s4 = new String("abc");//可以看出==比较的是栈的地址是否相同System.out.print("String类型用new String()是s1==s2:"); System.out.println(s3 == s4); System.out.println(s1 == s3); System.out.println("-----"); Integer i1 = 1; Integer i2 = 1; System.out.print("包装类型是i1==i2:"); System.out.println(i1 == i2); System.out.println("-----"); Integer i3 = 128; Integer i4 = 128;//此时输出false是因为Integer在-128-127之间会缓存,超出这个范围就不会缓存了System.out.print("包装类型是i3==i4:"); System.out.println(i3 == i4); System.out.println("-----"); Integer i5 = new Integer("1"); Integer i6 = new Integer("1"); System.out.print("包装类型用new Integer()是i5==i6:"); System.out.println(i5 == i6);//用new Integer()多少都不会缓存System.out.println("-----"); A a1 = new A(1); A a2 = new A(1); A a3 = a2; System.out.print("普通引用类型a1 == a2:"); System.out.println(a1 == a2); System.out.println(a2 == a3);//对象赋给新对象连地址都是相同的System.out.println("-----"); } }class A{int i;public A(int i){this.i = i; } }
基本类型a==b:true-----String类型是s1==s2:true-----String类型用new String()是s1==s2:falsefalse-----包装类型是i1==i2:true-----包装类型是i3==i4:false-----包装类型用new Integer()是i5==i6:false-----普通引用类型a1 == a2:falsetrue-----
package com.lk.C;public class Test8 {public static void main(String[] args) {// TODO Auto-generated method stubSystem.out.println("基本类型没有equals方法"); System.out.println("-----"); String s1 = "abc"; String s2 = "abc"; System.out.print("String类型的equals方法:"); System.out.println(s1.equals(s2)); System.out.println("-----"); String s3 = new String("abc"); String s4 = new String("abc");//可以看出比较equals方法比较的是堆里的值是否相同System.out.print("String类型的new String()的equals方法:"); System.out.println(s3.equals(s4)); System.out.println("-----"); System.out.print("String用==赋值和用new String()赋值的比较:"); System.out.println(s1.equals(s3)); System.out.println("-----"); Integer i1 = 1; Integer i2 = 1; System.out.print("包装类的equals方法:"); System.out.println(i1.equals(i2)); System.out.println("-----"); Integer i3 = new Integer(1); Integer i4 = new Integer(1); System.out.print("包装类的new Integer()用equals方法:"); System.out.println(i3.equals(i4)); System.out.println("-----"); System.out.print("Integer用==赋值和用new Integer()赋值的比较:"); System.out.println(i1.equals(i3)); System.out.println("-----"); } }
基本类型没有equals方法-----String类型的equals方法:true-----String类型的new String()的equals方法:true-----String用==赋值和用new String()赋值的比较:true-----包装类的equals方法:true-----包装类的new Integer()用equals方法:true-----Integer用==赋值和用new Integer()赋值的比较:true-----
package com.lk.C;public class Test9 {public static void main(String[] args) {// TODO Auto-generated method stubStudent s1 = new Student("阿坤",21); Student s2 = new Student("阿坤",21); Student s3 = new Student(); Student s4 = new Student(); Student s5 = s1; System.out.print("普通类对象的==非默认构造:"); System.out.println(s1 == s2); System.out.println(s1 == s5); System.out.println("-----"); System.out.print("普通类对象的equals非默认构造:"); System.out.println(s1.equals(s2)); System.out.println(s1.equals(s5)); System.out.println("-----"); System.out.print("普通类对象的==默认构造:"); System.out.println(s3 == s4); System.out.println("-----"); System.out.print("普通类对象的equals默认构造:"); System.out.println(s3.equals(s4)); System.out.println("-----"); System.out.print("对普通对象的属性进行比较equals:"); System.out.println(s1.name.equals(s2.name)); System.out.print("对普通对象的属性进行比较==:"); System.out.println(s1.name == s2.name); } }class Student{public String name;public int age;public Student(){ }public Student(String name,int age){this.name = name;this.age = age; }public void test(){ System.out.println(this.name); System.out.println(this.age); } }
普通类对象的==非默认构造:falsetrue-----普通类对象的equals非默认构造:falsetrue-----普通类对象的==默认构造:false-----普通类对象的equals默认构造:false-----对普通对象的属性进行比较equals:true对普通对象的属性进行比较==:true
从以上的三个程序可以看出:
1)对于==:在简单类型中(int等),这能使用该方法进行比较,这种类型没有equals方法,int的值是存在栈中的,==比较的是栈的内容是否相同。在String类型中,比较特殊,用String=“”;这种进行赋值时,两个相同的值用==比较也是相同的。但是用new String(),赋值就不相同。说明String=“”时,java会检查在堆中是否由相同的值,如果有,把新对象的地址也同老对象的地址赋为相同,因此==比较会相同。但是new String()开辟的就是两个栈,因此用==比较不会相同。对于包装类,如Integer=“”;时,在-128-127会有缓存,请看上面程序。其他的情况与String类似。
2)对于equals:当时String类型或者是包装类,如Integer时,比较的就是堆中的值,Integer也无缓存之说。对于普通类,equals比较的内存的首地址,这时候和==是一样的,即比较两边指向的是不是同一个对象。详细请见程序三。
很好,很详细的文章,感谢网友的分享,记录下来只为学习。
以上程序都是亲自测试过。希望能对大家有帮助。
以下是一些在百度中找到的说法:
java中, (1)对于字符串变量来说,equal比较的两边对象的内容,所以内容相同返回的是true。 至于你没问到的“==”,比较的是内存中的首地址,所以如果不是同一个对象,“==”不会返回true 而是false。 举个简单的例子, String s1="abc", s2="abc"; String s3 =new String("abc"); String s4=new String("abc"); s1==s2 //true,s1.equals(s2) //true,s3.equals(s3) //true,equal比较的是内容s3==s4//false,==比较的是首地址,所以是false(2)对于非字符串变量,equals比较的内存的首地址,这时候和==是一样的,即比较两边指向的是不是同一个对象, 即 Sample sa1 = new Sample(); Sample sa2 = new Sample(); sa1.equals(sa2) //false,因为不是同一对象 注意,如果加上 sa1=sa2; 那么 sa1.equals(sa2) //true
以上是為什麼equals()方法要重寫?的詳細內容。更多資訊請關注PHP中文網其他相關文章!