您好,登錄后才能下訂單哦!
今天就跟大家聊聊有關什么是Java中equals和==?有什么區別?,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結了以下內容,希望大家根據這篇文章可以有所收獲。
一、java當中的數據類型和“==”的含義:
基本數據類型(也稱原始數據類型) :byte,short,char,int,long,float,double,boolean。他們之間的比較,應用雙等號(==),比較的是他們的值。
引用數據類型:當他們用(==)進行比較的時候,比較的是他們在內存中的存放地址(確切的說,是堆內存地址)。
注:對于第二種類型,除非是同一個new出來的對象,他們的比較后的結果為true,否則比較后結果為false。因為每new一次,都會重新開辟堆內存空間。
二、equals()方法介紹:
JAVA當中所有的類都是繼承于Object這個超類的,在Object類中定義了一個equals的方法,equals的源碼是這樣寫的:
public boolean equals(Object obj) { //this - s1 //obj - s2 return (this == obj); }
可以看到,這個方法的初始默認行為是比較對象的內存地址值,一般來說,意義不大。所以,在一些類庫當中這個方法被重寫了,如String、Integer、Date。在這些類當中equals有其自身的實現(一般都是用來比較對象的成員變量值是否相同),而不再是比較類在堆內存中的存放地址了。
所以說,對于復合數據類型之間進行equals比較,在沒有覆寫equals方法的情況下,他們之間的比較還是內存中的存放位置的地址值,跟雙等號(==)的結果相同;如果被復寫,按照復寫的要求來。
我們對上面的兩段內容做個總結吧:
== 的作用:
基本類型:比較的就是值是否相同
引用類型:比較的就是地址值是否相同
equals 的作用:
引用類型:默認情況下,比較的是地址值。
注:不過,我們可以根據情況自己重寫該方法。一般重寫都是自動生成,比較對象的成員變量值是否相同
三、String類的equals()方法:
現在我們拿String類來舉例:
我們去\src\java\lang目錄中找到String類,發現equals方法被重寫如下:
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; }
上述代碼可以看出,String類中被復寫的equals()方法其實是比較兩個字符串的內容。下面我們通過實際代碼來看看String類的比較。
1、舉例代碼如下:
public class StringDemo { public static void main(String[] args) { String s1 = "Hello"; String s2 = "Hello"; System.out.println(s1 == s2); // true } }
上方代碼中,用“==”比較s1和s2,返回的結果是true。
2、稍微改動一下程序,會有奇怪的發現:
public class StringDemo { public static void main(String args[]) { String str1 = "Hello"; String str2 = new String("Hello"); String str3 = str2; // 引用傳遞 System.out.println(str1 == str2); // false System.out.println(str1 == str3); // false System.out.println(str2 == str3); // true System.out.println(str1.equals(str2)); // true System.out.println(str1.equals(str3)); // true System.out.println(str2.equals(str3)); // true } }
上方第4行代碼中,我們new了一個對象,用“==”比較s1和s2,返回的結果卻是false;而用用“equals”比較s1和s2,返回的結果是true。
為了分析上面的代碼,我們必須首先分析堆內存空間和棧內存空間,這一點非常重要:
請解釋字符串比較之中“==”和equals()的區別?
==:比較的是兩個字符串內存地址(堆內存)的數值是否相等,屬于數值比較;
equals():比較的是兩個字符串的內容,屬于內容比較。
以后進行字符串相等判斷的時候都使用equals()。
3、再次更改程序:
public class ObjectDemo{ public static void main(String[] args) { String s1 = "Hello"; String s2 = new String("Hello"); s2 = s2.intern(); System.out.println(s1 == s2); // true System.out.println(s1.equals(s2)); // true } }
上述代碼的第5行中,java.lang.String的intern()方法"abc".intern()方法的返回值還是字符串"abc",表面上看起來好像這個方法沒什么用處。但實際上,它做了個小動作:檢查字符串池里是否存在"abc"這么一個字符串,如果存在,就返回池里的字符串;如果不存在,該方法會 把"abc"添加到字符串池中,然后再返回它的引用。
四、比較兩個對象的值:
代碼如下:
package com.smyh; public class ObjectDemo { public static void main(String args[]){ Student student1 = new Student("生命壹號",22,"成都"); Student student2 = new Student("生命壹號",22,"成都"); System.out.println(student1==student2); System.out.println(student1.equals(student2)); } } class Student { private String name; private int age; private String address; public Student(String name,int age,String address){ this.name = name; this.age = age; this.address = address; } //重寫Object類中的equals方法(比較兩個對象的值是否相等) public boolean equals(Object obj){ //為了提高效率:如果兩個內存地址相等,那么一定是指向同一個對內存中的對象,就無需比較兩個對象的屬性值(自己跟自己比,沒啥意義嘛) if(this==obj){ return true; } //為了提供程序的健壯性 //我先判斷一下,obj是不是學生的一個對象,如果是,再做向下轉型,如果不是,直接返回false。 //這個時候,我們要判斷的是對象是否是某個類的對象? //記住一個格式:對象名 instanceof 類名。表示:判斷該對象是否是該類的一個對象 if(!(obj instanceof Student)){ return false; } //如果是就繼續 Student s = (Student)obj;//強制轉換,即向下轉型(畢竟Object類型沒有具體的對象屬性) return this.name.equals(s.name) && this.age == s.age && this.address.equals(s.address);//判斷兩個對象的屬性值是否相等 } }
上述代碼中,首先判斷傳遞進來的對象與當前對象的地址是否相等,如果相等,則肯定是同一個堆內存中的對象。因為傳遞進來的參數是Object類型,所以任何對象都可以接收。一旦接收進來,就將Object類型的對象向下轉型,然后判斷具體的屬性值。
運行效果:
五、equals()的重寫規則
前面我們已經知道如何去重寫equals方法來實現我們自己的需求了,但是我們在重寫equals方法時,還是需要注意如下幾點規則的。
自反性。對于任何非null的引用值x,x.equals(x)應返回true。
對稱性。對于任何非null的引用值x與y,當且僅當:y.equals(x)返回true時,x.equals(y)才返回true。
傳遞性。對于任何非null的引用值x、y與z,如果y.equals(x)返回true,y.equals(z)返回true,那么x.equals(z)也應返回true。
一致性。對于任何非null的引用值x與y,假設對象上equals比較中的信息沒有被修改,則多次調用x.equals(y)始終返回true或者始終返回false。
對于任何非空引用值x,x.equal(null)應返回false。
六、為什么重寫equals()的同時還得重寫hashCode()
我們知道equals()和hashCode()是java Object中兩個基本方法,有時候由于業務的需求,需要我們重寫equals()方法,比如對于Person類中,業務的需要讓我們當Person對象的cardID一致的時候,就認為兩個對象equals,此時就需要在Person類中重寫equals()方法,如下:
public class Person { public String name; public int age; public String cardID; .....// 省略 @Override public boolean equals(Object o) { if (o instanceof Person) { Person p = (Person) o; return this.cardID.equals(p.cardID); } else { return false; } } @Override public int hashCode() { return this.cardID.hashCode(); } ......// 省略 }
保證相同對象的hashCode一定相同,不同對象的hashCode基本相同。
閱讀HashMap的源碼,我們可以看到,HashMap中實現了一個Entry[]數組,數組的每個item是一個單項鏈表的結構,當我們put(key, value)的時候,HashMap首先會newItem.key.hashCode()作為該newItem在Entry[]中存儲的下標,要是對應的下標的位置上沒有任何item,則直接存儲上去,要是已經有oldItem存儲在了上面,那就會判斷oldItem.key.equals(newItem.key),那么要是我們把上面的Person作為key進行存儲的時候,重寫了equals()方法,但沒重寫hashCode()方法,當我們去put()的時候,首先會通過hashCode() 計算下標,由于沒有重寫hashCode(),那么實質是完整的Object的hashCode(),會受到Object多個屬性的影響,本來equals的兩個Person對象,反而得到了兩個不同的下標。
同樣的,HashMap在get(key)的過程中,也是首先調用hashCode()計算item的下標,然后在對應下標的地方找,要是為null,就返回null,要是 != null,會去調用equals()方法,比較key是否一致,只有當key一致的時候,才會返回value,要是我們沒有重寫hashCode()方法,本來有的item,反而會找不到,返回null結果。
所以,要是你重寫了equals()方法,而你的對象可能會放入到散列(HashMap,HashTable或HashSet等)中,那么還必須重寫hashCode(), 如果你的對象有可能放到有序隊列(實現了Comparable)中,那么還需要重寫compareTo()的方法。
看完上述內容,你們對什么是Java中equals和==?有什么區別?有進一步的了解嗎?如果還想了解更多知識或者相關內容,請關注億速云行業資訊頻道,感謝大家的支持。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。