Java之"=="和equals的區別

轉自:http://blog.csdn.net/qq_33406883/article/details/52180154
一、java當中的數據類型和“==”的含義:

基本數據類型(也稱原始數據類型) :byte,short,char,int,long,float,double,boolean。他們之間的比較,應用雙等號(==),比較的是他們的值。
複合數據類型(類):當他們用(==)進行比較的時候,比較的是他們在內存中的存放地址(確切的說,是堆內存地址)。
注:對於第二種類型,除非是同一個new出來的對象,他們的比較後的結果爲true,否則比較後結果爲false。因爲每new一次,都會重新開闢堆內存空間。

二、equals()方法介紹:

JAVA當中所有的類都是繼承於Object這個超類的,在Object類中定義了一個equals的方法,這個方法的初始行爲是比較對象的內存地址,但在一些類庫當中這個方法被複寫了,如String、Integer、Date。在這些類當中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); // true
6 }
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)); // true
10 System.out.println(str1.equals(str3)); // true
11 System.out.println(str2.equals(str3)); // true
12 }
13 }

上方第4行代碼中,我們new了一個對象,用“==”比較s1和s2,返回的結果卻是false;而用用“equals”比較s1和s2,返回的結果是true。
爲了分析上面的代碼,我們必須首先分析堆內存空間和棧內存空間,這一點非常重要.

面試題:請解釋字符串比較之中“==”和equals()的區別?

==:比較的是兩個字符串內存地址的數值是否相等,屬於數值比較;
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); // true
7 System.out.println(s1.equals(s2)); // true
8 }
9 }

上述代碼的第5行中,java.lang.String的intern()方法”abc”.intern()方法的返回值還是字符串”abc”,表面上看起來好像這個方法沒什麼用處。但實際上,它做了個小動作:檢查字符串池裏是否存在”abc”這麼一個字符串,如果存在,就返回池裏的字符串;如果不存在,該方法會 把”abc”添加到字符串池中,然後再返回它的引用。

四、比較兩個對象的值:

代碼如下:

1 public class ObjectDemo {
2 public static void main(String args[]){
3 Student student1 = new Student(“生命壹號”,22,”成都”);
4 Student student2 = new Student(“生命壹號”,22,”成都”);
5 System.out.println(student1==student2);
6 System.out.println(student1.equals(student2));
7 }
8 }
9 class Student {
10 private String name;
11 private int age;
12 private String address;
13 public Student(String name,int age,String address){
14 this.name = name;
15 this.age = age;
16 this.address = address;
17 }
18 //重寫Object類中的equals方法(比較兩個對象的值是否相等)
19 public boolean equals(Object obj){
20 //如果內存地址相等,那麼一定是同一個對象,就無需比較兩個對象的屬性值
21 if(this==obj){
22 return true;
23 }
24 //判斷obj是否爲Baboon類型的實例
25 if(obj instanceof Student){
26 Student b = (Student)obj;//強制轉換
27 //判斷兩個對象的屬性值是否相等
28 if(!this.name.equals(b.name)){
29 return false;
30 }
31 else if(this.age!=b.age){
32 return false;
33 }
34 else if(this.address!=b.address){
35 return false;
36 }
37 return true;
38 }else{
39 return false;
40 }
41 }
42 }

上述代碼中,首先判斷傳遞進來的對象與當前對象的地址是否相等,如果相等,則肯定是同一個對象。因爲傳遞進來的參數是Object類型,所以任何對象都可以接收。一旦接收進來,就將Object類型的對象向下轉型,然後再做判斷。

public class EqualTest {
public static void main(String[] args) {

//對於基本類型的變量。"=="和"equal"的區別 
int t1=57; 
int t2=67; 
int t3=124; 
int t4=124; 

//“==”對於基本數據類型,判斷兩個變量的值是否相等。 
Boolean result1=(t1==t2); 
Boolean result2=((t1+t2)==t3); 
Boolean result3=(t3==t4); 

System.out.println("/n/n-----【t1==t2】"+result1+"/n-----【(t1+t2)=t3】"+result2+"/n-----【t3=t4】"+result3); 
//“equal”不能用於基本數據類型。只能用於類變量。對於基本數據類型要用其包裝類。 
Integer i1=new Integer(t1); 
Integer i2=new Integer(t2); 
Integer i3=new Integer(t3); 
Integer i4=new Integer(t4); 


Boolean ri1=i1.equals(i2); 
Boolean ri2=i3.equals(i1+i2); 
Boolean ri3=i3.equals(i4); 

System.out.println("/n/n-----【i1.equals(i2)】"+ri1+"/n-----【i3.equals(i1+i2)】"+ri2+"/n-----【i3.equals(i4)】"+ri3); 

//對於對象變量,"=="和"equal"的區別 

String st1="wasiker "; 
String st2="is super man"; 
String st3="wasiker is super man"; 
String st4="wasiker is super man"; 

Boolean b1=(st1==st2); 
Boolean b2=(st1+st2)==st3; 
Boolean b3=(st3==st4); 

System.out.println("/n/n-----【st1==st2】"+b1+"/n-----【(st1+st2)==st3】"+b2+"/n-----【st3==st4】"+b3); 

//因爲對象變量的存儲的是對象在內存中的路徑,即內存地址。所以用“==”比較時,即使
//對象的值相等,但是他們的內存地址不同,所以==的結果爲false。故“==”用於比較兩
//個變量的值是否相等,而不是變量引用的對象是否相等

Boolean r1=st1.equals(st2); 
Boolean r2=(st1+st2).equals(st3); 
Boolean r3=st3.equals(st4); 

System.out.println("/n/n-----【st1.equals(st2)】"+r1+"/n-----【(st1+st2).equals(st3)】"+r2+"/n-----【st3.equals(st4)】"+r3); 

//equal用於比較兩個對象是否相同。
}
}
運行結果爲:
—–【t1==t2】false
—–【(t1+t2)=t3】true
—–【t3=t4】true

—–【i1.equals(i2)】false
—–【i3.equals(i1+i2)】true
—–【i3.equals(i4)】true

—–【st1==st2】false
—–【(st1+st2)==st3】false
—–【st3==st4】true

—–【st1.equals(st2)】false
—–【(st1+st2).equals(st3)】true
—–【st3.equals(st4)】true

總之:
“==”比較的是值【變量(棧)內存中存放的對象的(堆)內存地址】
equal用於比較兩個對象的值是否相同【不是比地址】

【特別注意】Object類中的equals方法和“==”是一樣的,沒有區別,而String類,Integer類等等一些類,是重寫了equals方法,才使得equals和“==不同”,所以,當自己創建類時,自動繼承了Object的equals方法,要想實現不同的等於比較,必須重寫equals方法。

“==”比”equal”運行速度快,因爲”==”只是比較引用.

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章