java提升篇(十三)-----equals()方法總結

equals()    

      超類Object中有這個equals()方法,該方法主要用於比較兩個對象是否相等。該方法的源碼以下:java

public boolean equals(Object obj) {
    return (this == obj);
    }

      咱們知道全部的對象都擁有標識(內存地址)和狀態(數據),同時「==」比較兩個對象的的內存地址,因此說使用Object的equals()方法是比較兩個對象的內存地址是否相等,即若object1.equals(object2)爲true,則表示equals1和equals2其實是引用同一個對象。雖然有時候Object的equals()方法能夠知足咱們一些基本的要求,可是咱們必需要清楚咱們很大部分時間都是進行兩個對象的比較,這個時候Object的equals()方法就不能夠了,實際上JDK中,String、Math等封裝類都對equals()方法進行了重寫。下面是String的equals()方法:數組

public boolean equals(Object anObject) {
    if (this == anObject) {
        return true;
    }
    if (anObject instanceof String) {
        String anotherString = (String)anObject;
        int n = count;
        if (n == anotherString.count) {
        char v1[] = value;
        char v2[] = anotherString.value;
        int i = offset;
        int j = anotherString.offset;
        while (n-- != 0) {
            if (v1[i++] != v2[j++])
            return false;
        }
        return true;
        }
    }
    return false;
    }

      對於這個代碼段:if (v1[i++] != v2[j++])return false;咱們能夠很是清晰的看到String的equals()方法是進行內容比較,而不是引用比較。至於其餘的封裝類都差很少。安全

      在Java規範中,它對equals()方法的使用必需要遵循以下幾個規則:測試

      equals 方法在非空對象引用上實現相等關係: this

     一、自反性:對於任何非空引用值 x,x.equals(x) 都應返回 true。 spa

     二、對稱性:對於任何非空引用值 x 和 y,當且僅當 y.equals(x) 返回 true 時,x.equals(y) 才應返回 true。 code

      三、傳遞性:對於任何非空引用值 x、y 和 z,若是 x.equals(y) 返回 true,而且 y.equals(z) 返回 true,那麼 x.equals(z) 應返回 true。 對象

      四、一致性:對於任何非空引用值 x 和 y,屢次調用 x.equals(y) 始終返回 true 或始終返回 false,前提是對象上 equals 比較中所用的信息沒有被修改。 blog

     五、 對於任何非空引用值 x,x.equals(null) 都應返回 false。  繼承

      對於上面幾個規則,咱們在使用的過程當中最好遵照,不然會出現意想不到的錯誤。

      在java中進行比較,咱們須要根據比較的類型來選擇合適的比較方式:

     1) 對象域,使用equals方法 。
       2) 類型安全的枚舉,使用equals或== 。
      3) 可能爲null的對象域 : 使用 == 和 equals 。
     4) 數組域 : 使用 Arrays.equals 。
     5) 除float和double外的原始數據類型 : 使用 == 。
     6) float類型: 使用Float.foatToIntBits轉換成int類型,而後使用==。 
      7) double類型: 使用Double.doubleToLongBit轉換成long類型,而後使用==。

至於6)、7)爲何須要進行轉換,咱們能夠參考他們相應封裝類的equals()方法,下面的是Float類的:

public boolean equals(Object obj) {
    return (obj instanceof Float)
           && (floatToIntBits(((Float)obj).value) == floatToIntBits(value));
    }

緣由嘛,裏面提到了兩點:

However, there are two exceptions:
If f1 and f2 both represent
Float.NaN, then the equals method returns
true, even though Float.NaN==Float.NaN
has the value false.
If <code>f1 represents +0.0f while
f2 represents -0.0f, or vice
versa, the equal test has the value
false, even though 0.0f==-0.0f
has the value true.

      在equals()中使用getClass進行類型判斷

      咱們在覆寫equals()方法時,通常都是推薦使用getClass來進行類型判斷,不是使用instanceof。咱們都清楚instanceof的做用是判斷其左邊對象是否爲其右邊類的實例,返回boolean類型的數據。能夠用來判斷繼承中的子類的實例是否爲父類的實現。注意後面這句話:能夠用來判斷繼承中的子類的實例是否爲父類的實現,正是這句話在做怪。咱們先看以下實例(摘自《高質量代碼 改善java程序的151個建議》)。

       父類:Person

public class Person {
    protected String name;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
    
    public Person(String name){
        this.name = name;
    }
    
    public boolean equals(Object object){
        if(object instanceof Person){
            Person p = (Person) object;
            if(p.getName() == null || name == null){
                return false;
            }
            else{
                return name.equalsIgnoreCase(p.getName());
            }
        }
        return false;
    }
}

      子類:Employee

public class Employee extends Person{
    private int id;
    
    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public Employee(String name,int id){
        super(name);
        this.id = id;
    }
    
    /**
     * 重寫equals()方法
     */
    public boolean equals(Object object){
        if(object instanceof Employee){
            Employee e = (Employee) object;
            return super.equals(object) && e.getId() == id;
        }
        return false;
    }
}

     上面父類Person和子類Employee都重寫了equals(),不過Employee比父類多了一個id屬性。測試程序以下:

public class Test {
    public static void main(String[] args) {
        Employee e1 = new Employee("chenssy", 23);
        Employee e2 = new Employee("chenssy", 24);
        Person p1 = new Person("chenssy");
        
        System.out.println(p1.equals(e1));
        System.out.println(p1.equals(e2));
        System.out.println(e1.equals(e2));
    }
}

      上面定義了兩個員工和一個普通人,雖然他們同名,可是他們確定不是同一人,因此按理來講輸出結果應該所有都是false,可是事與願違,結果是:true、true、false。

      對於那e1!=e2咱們很是容易理解,由於他們不只須要比較name,還須要比較id。可是p1即等於e1也等於e2,這是很是奇怪的,由於e一、e2明明是兩個不一樣的類,但爲何會出現這個狀況?首先p1.equals(e1),是調用p1的equals方法,該方法使用instanceof關鍵字來檢查e1是否爲Person類,這裏咱們再看看instanceof:判斷其左邊對象是否爲其右邊類的實例,也能夠用來判斷繼承中的子類的實例是否爲父類的實現。他們二者存在繼承關係,確定會返回true了,而二者name又相同,因此結果確定是true。

      因此出現上面的狀況就是使用了關鍵字instanceof,這是很是容易「專空子」的。故在覆寫equals時推薦使用getClass進行類型判斷。而不是使用instanceof。

鞏固基礎,提升技術,不懼困難,攀登高峯!!!!!!

相關文章
相關標籤/搜索