【聲明】
歡迎轉載,但請保留文章原始出處→_→
生命壹號:http://www.companysz.com/smyhvae/
文章來源:http://www.companysz.com/smyhvae/p/3929585.html
聯系方式:[email protected]
【正文】
平時在學Android和Java語言的時候,總是碰到“equals”和“==”這兩個字符,老感覺差不多;其實還是有一些區別的,今天干脆把它們徹底弄清楚。
一、java當中的數據類型和“==”的含義:
注:對于第二種類型,除非是同一個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方法被重寫如下:
1 public boolean equals(Object anObject) { 2 if (this == anObject) { 3 return true; 4 } 5 if (anObject instanceof String) { 6 String anotherString = (String)anObject; 7 int n = value.length; 8 if (n == anotherString.value.length) { 9 char v1[] = value;10 char v2[] = anotherString.value;11 int i = 0;12 while (n-- != 0) {13 if (v1[i] != v2[i])14 return false;15 i++;16 }17 return true;18 }19 }20 return false;21 }
上述代碼可以看出,String類中被復寫的equals()方法其實是比較兩個字符串的內容。下面我們通過實際代碼來看看String類的比較。
1、舉例代碼如下:
1 public class StringDemo {2 public static void main(String[] args) {3 String s1 = "Hello";4 String s2 = "Hello";5 System.out.PRintln(s1 == s2); // true6 }7 }
上方代碼中,用“==”比較s1和s2,返回的結果是true。
2、稍微改動一下程序,會有奇怪的發現:
1 public class StringDemo { 2 public static void main(String args[]) { 3 String str1 = "Hello"; 4 String str2 = new String("Hello"); 5 String str3 = str2; // 引用傳遞 6 System.out.println(str1 == str2); // false 7 System.out.println(str1 == str3); // false 8 System.out.println(str2 == str3); // true 9 System.out.println(str1.equals(str2)); // true10 System.out.println(str1.equals(str3)); // true11 System.out.println(str2.equals(str3)); // true12 }13 }
上方第4行代碼中,我們new了一個對象,用“==”比較s1和s2,返回的結果卻是false;而用用“equals”比較s1和s2,返回的結果是true。
為了分析上面的代碼,我們必須首先分析堆內存空間和棧內存空間,這一點非常重要:
看完上面的圖,再結合上面的代碼,就一目了然了。現在我們可以給自己出一道面試題:
面試題:請解釋字符串比較之中“==”和equals()的區別?
以后進行字符串相等判斷的時候都使用equals()。
3、再次更改程序:
1 public class ObjectDemo{2 public static void main(String[] args) {3 String s1 = "Hello";4 String s2 = new String("Hello");5 s2 = s2.intern();6 System.out.println(s1 == s2); // true7 System.out.println(s1.equals(s2)); // true8 }9 }
上述代碼的第5行中,java.lang.String的intern()方法"abc".intern()方法的返回值還是字符串"abc",表面上看起來好像這個方法沒什么用處。但實際上,它做了個小動作:檢查字符串池里是否存在"abc"這么一個字符串,如果存在,就返回池里的字符串;如果不存在,該方法會 把"abc"添加到字符串池中,然后再返回它的引用。
四、比較兩個對象的值:
代碼如下:
1 package com.smyh; 2 3 public class ObjectDemo { 4 public static void main(String args[]){ 5 Student student1 = new Student("生命壹號",22,"成都"); 6 Student student2 = new Student("生命壹號",22,"成都"); 7 System.out.println(student1==student2); 8 System.out.println(student1.equals(student2)); 9 }10 }11 class Student {12 private String name;13 private int age;14 private String address;15 public Student(String name,int age,String address){16 this.name = name;17 this.age = age;18 this.address = address;19 }20 //重寫Object類中的equals方法(比較兩個對象的值是否相等)21 public boolean equals(Object obj){22 //為了提高效率:如果兩個內存地址相等,那么一定是指向同一個對內存中的對象,就無需比較兩個對象的屬性值(自己跟自己比,沒啥意義嘛)23 if(this==obj){24 return true;25 }26 27 //為了提供程序的健壯性28 //我先判斷一下,obj是不是學生的一個對象,如果是,再做向下轉型,如果不是,直接返回false。29 //這個時候,我們要判斷的是對象是否是某個類的對象?30 //記住一個格式:對象名 instanceof 類名。表示:判斷該對象是否是該類的一個對象 31 if(!(obj instanceof Student)){ 32 return false; 33 }34 35 //如果是就繼續36 Student s = (Student)obj;//強制轉換,即向下轉型(畢竟Object類型沒有具體的對象屬性) 37 return this.name.equals(s.name) && this.age == s.age && this.address.equals(s.address);//判斷兩個對象的屬性值是否相等38 }39 }
上述代碼中,首先判斷傳遞進來的對象與當前對象的地址是否相等,如果相等,則肯定是同一個堆內存中的對象。因為傳遞進來的參數是Object類型,所以任何對象都可以接收。一旦接收進來,就將Object類型的對象向下轉型,然后判斷具體的屬性值。
運行效果:
其實,如果是在Eclipse中做開發的話,上面重寫的equals()方法其實是可以自動生成的:
自動生成后的equals()方法如下:
1 @Override 2 public boolean equals(Object obj) { 3 if (this == obj) 4 return true; 5 if (obj == null) 6 return false; 7 if (getClass() != obj.getClass()) 8 return false; 9 Student other = (Student) obj;10 if (address == null) {11 if (other.address != null)12 return false;13 } else if (!address.equals(other.address))14 return false;15 if (age != other.age)16 return false;17 if (name == null) {18 if (other.name != null)19 return false;20 } else if (!name.equals(other.name))21 return false;22 return true;23 }
可以看到:我們自己重寫的equals()方法和自動生成的equals()方法,二者效果是一樣的。
【參考】
參考博客:
http://www.companysz.com/zhxhdean/archive/2011/03/25/1995431.html
http://www.companysz.com/zhxhdean/archive/2011/03/26/1996468.html
以及李興華老師的視頻講解。
新聞熱點
疑難解答