Java equals ==

來源:互聯網
上載者:User

標籤:相同   color   標識   ted   repr   參考   another   記憶體位址   修改   

package com.cwcec.test;class Demo{    }class MyDemo extends Demo{    }public class FieldDemo {    public static void main(String[] args)    {                 MyDemo mydemo = new MyDemo();        Object object = mydemo;        Demo demo = mydemo;                System.out.println(object instanceof MyDemo);                  System.out.println(mydemo instanceof Demo);        System.out.println(demo instanceof MyDemo);                    System.out.println(demo.equals(mydemo));        System.out.println(mydemo.equals(demo));                String s = "123";        String s1 = new String("123");                System.out.println(s == "123");        System.out.println(s1 == "123");        System.out.println(s == s1);        System.out.println(s.equals(s1));            }  } 
Output: true true true true true true false false true
equals()    

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

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 方法在非Null 物件引用上實現相等關係:

     1、自反性:對於任何非Null 參考值 x,x.equals(x) 都應返回 true。

     2、對稱性:對於任何非Null 參考值 x 和 y,若且唯若 y.equals(x) 返回 true 時,x.equals(y) 才應返回 true。

      3、傳遞性:對於任何非Null 參考值 x、y 和 z,如果 x.equals(y) 返回 true,並且 y.equals(z) 返回 true,那麼 x.equals(z) 應返回 true。

      4、一致性:對於任何非Null 參考值 x 和 y,多次調用 x.equals(y) 始終返回 true 或始終返回 false,前提是對象上 equals 比較中所用的資訊沒有被修改。

     5、 對於任何非Null 參考值 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 representFloat.NaN, then the equals method returnstrue, even though Float.NaN==Float.NaNhas the value false.If <code>f1 represents +0.0f whilef2 represents -0.0f, or viceversa, the equal test has the valuefalse, even though 0.0f==-0.0fhas 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,這是非常奇怪的,因為e1、e2明明是兩個不同的類,但為什麼會出現這個情況?首先p1.equals(e1),是調用p1的equals方法,該方法使用instanceof關鍵字來檢查e1是否為Person類,這裡我們再看看instanceof:判斷其左邊對象是否為其右邊類的執行個體,也可以用來判斷繼承中的子類的執行個體是否為父類的實現。他們兩者存在繼承關係,肯定會返回true了,而兩者name又相同,所以結果肯定是true。

      所以出現上面的情況就是使用了關鍵字instanceof,這是非常容易“專空子”的。故在覆寫equals時推薦使用getClass進行類型判斷。而不是使用instanceof。

Java equals ==

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.