溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

什么是Java中equals和==?有什么區(qū)別?

發(fā)布時(shí)間:2020-07-14 11:00:05 來源:億速云 閱讀:140 作者:Leah 欄目:編程語言

今天就跟大家聊聊有關(guān)什么是Java中equals和==?有什么區(qū)別?,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。

一、java當(dāng)中的數(shù)據(jù)類型和“==”的含義:

  • 基本數(shù)據(jù)類型(也稱原始數(shù)據(jù)類型) :byte,short,char,int,long,float,double,boolean。他們之間的比較,應(yīng)用雙等號(hào)(==),比較的是他們的值。

  • 引用數(shù)據(jù)類型:當(dāng)他們用(==)進(jìn)行比較的時(shí)候,比較的是他們?cè)趦?nèi)存中的存放地址(確切的說,是堆內(nèi)存地址)。

注:對(duì)于第二種類型,除非是同一個(gè)new出來的對(duì)象,他們的比較后的結(jié)果為true,否則比較后結(jié)果為false。因?yàn)槊縩ew一次,都會(huì)重新開辟堆內(nèi)存空間。

二、equals()方法介紹:

JAVA當(dāng)中所有的類都是繼承于Object這個(gè)超類的,在Object類中定義了一個(gè)equals的方法,equals的源碼是這樣寫的:

public boolean equals(Object obj) {
    //this - s1
    //obj - s2
    return (this == obj);
}

可以看到,這個(gè)方法的初始默認(rèn)行為是比較對(duì)象的內(nèi)存地址值,一般來說,意義不大。所以,在一些類庫當(dāng)中這個(gè)方法被重寫了,如String、Integer、Date。在這些類當(dāng)中equals有其自身的實(shí)現(xiàn)(一般都是用來比較對(duì)象的成員變量值是否相同),而不再是比較類在堆內(nèi)存中的存放地址了。

所以說,對(duì)于復(fù)合數(shù)據(jù)類型之間進(jìn)行equals比較,在沒有覆寫equals方法的情況下,他們之間的比較還是內(nèi)存中的存放位置的地址值,跟雙等號(hào)(==)的結(jié)果相同;如果被復(fù)寫,按照復(fù)寫的要求來。

我們對(duì)上面的兩段內(nèi)容做個(gè)總結(jié)吧:

== 的作用:
基本類型:比較的就是值是否相同
引用類型:比較的就是地址值是否相同

equals 的作用:
引用類型:默認(rèn)情況下,比較的是地址值。

注:不過,我們可以根據(jù)情況自己重寫該方法。一般重寫都是自動(dòng)生成,比較對(duì)象的成員變量值是否相同

三、String類的equals()方法:

現(xiàn)在我們拿String類來舉例:

我們?nèi)src\java\lang目錄中找到String類,發(fā)現(xiàn)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類中被復(fù)寫的equals()方法其實(shí)是比較兩個(gè)字符串的內(nèi)容。下面我們通過實(shí)際代碼來看看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,返回的結(jié)果是true。

2、稍微改動(dòng)一下程序,會(huì)有奇怪的發(fā)現(xiàn):

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了一個(gè)對(duì)象,用“==”比較s1和s2,返回的結(jié)果卻是false;而用用“equals”比較s1和s2,返回的結(jié)果是true。

為了分析上面的代碼,我們必須首先分析堆內(nèi)存空間和棧內(nèi)存空間,這一點(diǎn)非常重要:

什么是Java中equals和==?有什么區(qū)別?

請(qǐng)解釋字符串比較之中“==”和equals()的區(qū)別?

  • ==:比較的是兩個(gè)字符串內(nèi)存地址(堆內(nèi)存)的數(shù)值是否相等,屬于數(shù)值比較;

  • equals():比較的是兩個(gè)字符串的內(nèi)容,屬于內(nèi)容比較。

以后進(jìn)行字符串相等判斷的時(shí)候都使用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",表面上看起來好像這個(gè)方法沒什么用處。但實(shí)際上,它做了個(gè)小動(dòng)作:檢查字符串池里是否存在"abc"這么一個(gè)字符串,如果存在,就返回池里的字符串;如果不存在,該方法會(huì) 把"abc"添加到字符串池中,然后再返回它的引用。

四、比較兩個(gè)對(duì)象的值:

代碼如下:

package com.smyh;

public class ObjectDemo {
    public static void main(String args[]){
        Student student1 = new Student("生命壹號(hào)",22,"成都");
        Student student2 = new Student("生命壹號(hào)",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方法(比較兩個(gè)對(duì)象的值是否相等)
    public boolean equals(Object obj){
        //為了提高效率:如果兩個(gè)內(nèi)存地址相等,那么一定是指向同一個(gè)對(duì)內(nèi)存中的對(duì)象,就無需比較兩個(gè)對(duì)象的屬性值(自己跟自己比,沒啥意義嘛)
        if(this==obj){
            return true;
        }
        
        //為了提供程序的健壯性
        //我先判斷一下,obj是不是學(xué)生的一個(gè)對(duì)象,如果是,再做向下轉(zhuǎn)型,如果不是,直接返回false。
        //這個(gè)時(shí)候,我們要判斷的是對(duì)象是否是某個(gè)類的對(duì)象?
        //記住一個(gè)格式:對(duì)象名 instanceof 類名。表示:判斷該對(duì)象是否是該類的一個(gè)對(duì)象       
        if(!(obj instanceof Student)){        
            return false;                
        }
        
        //如果是就繼續(xù)
        Student s = (Student)obj;//強(qiáng)制轉(zhuǎn)換,即向下轉(zhuǎn)型(畢竟Object類型沒有具體的對(duì)象屬性)        
        return this.name.equals(s.name) && this.age == s.age && this.address.equals(s.address);//判斷兩個(gè)對(duì)象的屬性值是否相等
    }
}

上述代碼中,首先判斷傳遞進(jìn)來的對(duì)象與當(dāng)前對(duì)象的地址是否相等,如果相等,則肯定是同一個(gè)堆內(nèi)存中的對(duì)象。因?yàn)閭鬟f進(jìn)來的參數(shù)是Object類型,所以任何對(duì)象都可以接收。一旦接收進(jìn)來,就將Object類型的對(duì)象向下轉(zhuǎn)型,然后判斷具體的屬性值。

運(yùn)行效果:

什么是Java中equals和==?有什么區(qū)別?

五、equals()的重寫規(guī)則

前面我們已經(jīng)知道如何去重寫equals方法來實(shí)現(xiàn)我們自己的需求了,但是我們?cè)谥貙慹quals方法時(shí),還是需要注意如下幾點(diǎn)規(guī)則的。

  • 自反性。對(duì)于任何非null的引用值x,x.equals(x)應(yīng)返回true。

  • 對(duì)稱性。對(duì)于任何非null的引用值x與y,當(dāng)且僅當(dāng):y.equals(x)返回true時(shí),x.equals(y)才返回true。

  • 傳遞性。對(duì)于任何非null的引用值x、y與z,如果y.equals(x)返回true,y.equals(z)返回true,那么x.equals(z)也應(yīng)返回true。

  • 一致性。對(duì)于任何非null的引用值x與y,假設(shè)對(duì)象上equals比較中的信息沒有被修改,則多次調(diào)用x.equals(y)始終返回true或者始終返回false。

  • 對(duì)于任何非空引用值x,x.equal(null)應(yīng)返回false。

六、為什么重寫equals()的同時(shí)還得重寫hashCode()

我們知道equals()和hashCode()是java Object中兩個(gè)基本方法,有時(shí)候由于業(yè)務(wù)的需求,需要我們重寫equals()方法,比如對(duì)于Person類中,業(yè)務(wù)的需要讓我們當(dāng)Person對(duì)象的cardID一致的時(shí)候,就認(rèn)為兩個(gè)對(duì)象equals,此時(shí)就需要在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();
    }

    ......// 省略


}

保證相同對(duì)象的hashCode一定相同,不同對(duì)象的hashCode基本相同。

閱讀HashMap的源碼,我們可以看到,HashMap中實(shí)現(xiàn)了一個(gè)Entry[]數(shù)組,數(shù)組的每個(gè)item是一個(gè)單項(xiàng)鏈表的結(jié)構(gòu),當(dāng)我們put(key, value)的時(shí)候,HashMap首先會(huì)newItem.key.hashCode()作為該newItem在Entry[]中存儲(chǔ)的下標(biāo),要是對(duì)應(yīng)的下標(biāo)的位置上沒有任何item,則直接存儲(chǔ)上去,要是已經(jīng)有oldItem存儲(chǔ)在了上面,那就會(huì)判斷oldItem.key.equals(newItem.key),那么要是我們把上面的Person作為key進(jìn)行存儲(chǔ)的時(shí)候,重寫了equals()方法,但沒重寫hashCode()方法,當(dāng)我們?nèi)ut()的時(shí)候,首先會(huì)通過hashCode() 計(jì)算下標(biāo),由于沒有重寫hashCode(),那么實(shí)質(zhì)是完整的Object的hashCode(),會(huì)受到Object多個(gè)屬性的影響,本來equals的兩個(gè)Person對(duì)象,反而得到了兩個(gè)不同的下標(biāo)。

同樣的,HashMap在get(key)的過程中,也是首先調(diào)用hashCode()計(jì)算item的下標(biāo),然后在對(duì)應(yīng)下標(biāo)的地方找,要是為null,就返回null,要是 != null,會(huì)去調(diào)用equals()方法,比較key是否一致,只有當(dāng)key一致的時(shí)候,才會(huì)返回value,要是我們沒有重寫hashCode()方法,本來有的item,反而會(huì)找不到,返回null結(jié)果。

所以,要是你重寫了equals()方法,而你的對(duì)象可能會(huì)放入到散列(HashMap,HashTable或HashSet等)中,那么還必須重寫hashCode(), 如果你的對(duì)象有可能放到有序隊(duì)列(實(shí)現(xiàn)了Comparable)中,那么還需要重寫compareTo()的方法。

看完上述內(nèi)容,你們對(duì)什么是Java中equals和==?有什么區(qū)別?有進(jìn)一步的了解嗎?如果還想了解更多知識(shí)或者相關(guān)內(nèi)容,請(qǐng)關(guān)注億速云行業(yè)資訊頻道,感謝大家的支持。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI