App下載

Java是值傳遞還是引用傳遞?有圖為證

猿友 2020-09-02 14:30:20 瀏覽數(shù) (2335)
反饋

文章轉(zhuǎn)載自公眾號:Java中文社群 作者:磊哥

開篇先來曝答案,在 Java 語言中,本質(zhì)只有值傳遞,而無引用傳遞,解釋和證明詳見正文。

說到值傳遞和引用傳遞我們不得不提到兩個概念:值類型和引用類型。

1.值類型

通俗意義上來說,所謂的值類型指的就是 Java 中的 8 大基礎(chǔ)數(shù)據(jù)類型:

  • 整數(shù)型:byte、int、short、long
  • 浮點(diǎn)型:float、double
  • 字符類型:char
  • 布爾類型:boolean

Java中的8大基礎(chǔ)數(shù)據(jù)類型

從 JVM 層面來講:所謂的值類型指的是在賦值時,直接在棧中(Java 虛擬機(jī)棧)生成值的類型,如下圖所示:

Java是值傳遞還是引用傳遞?上圖為證

2.引用類型

引用類型是指除值類型之外的數(shù)據(jù)類型,比如:

  • 接口
  • 數(shù)組
  • 字符串
  • 包裝類(Integer、Double...)

Java是值傳遞還是引用傳遞?上圖為證

從 JVM 的層面來講,所謂的引用類型是指,在初始化時將引用生成棧上,而值生成在堆上的這些數(shù)據(jù)類型,如下圖所示:

Java是值傳遞還是引用傳遞?上圖為證

3.值傳遞

值傳遞(Pass By Value)指的是方法傳參時,傳遞的是原內(nèi)容的副本,因此對副本進(jìn)行如何修改都不會影響原內(nèi)容。

實(shí)現(xiàn)代碼如下:

public class PassTest {
    public static void main(String[] args) {
        int age = 18;
        System.out.println("調(diào)用方法前:" + age);
        intTest(age);
        System.out.println("調(diào)用方法后:" + age);
    }


    private static void intTest(int age) {
        age = 30;
        System.out.println("方法中修改為:" + age);
    }
}

程序的執(zhí)行結(jié)果為:

調(diào)用方法前:18

方法中修改為:30

調(diào)用方法后:18

從上述結(jié)果可以看出,在方法中修改參數(shù)并未影響原內(nèi)容,我們把這種傳參方式稱之為值傳遞。

4.引用傳遞

引用傳遞(Pass By Reference)指的是方法傳參時,傳遞的是參數(shù)本身,因此對參數(shù)進(jìn)行任意修改都會影響原內(nèi)容。

模擬“引用傳遞”的實(shí)現(xiàn)代碼如下:

public class PassTest {
    public static void main(String[] args) {
        char[] name = {'編程', '師'};
        System.out.println("調(diào)用方法前:" + new String(name));
        paramTest(name);
        System.out.println("調(diào)用方法后:" + new String(name));
    }
    private static void paramTest(char[] n) {
        n[1] = '獅';
        System.out.println("方法中修改為:" + new String(n));
    }
}

程序的執(zhí)行結(jié)果為:

調(diào)用方法前:編程師

方法中修改為:編程獅

調(diào)用方法后:編程獅

從上述的結(jié)果可以看出在 paramTest 方法中修改了參數(shù)之后,在 main 方法中再打印參數(shù)時,發(fā)現(xiàn)參數(shù)的值也跟著發(fā)生了改變,那么似乎我們可以得出結(jié)論,Java 中貌似也有“引用傳遞”,然而實(shí)事并如此,我們接著看。

5.真假“引用傳遞”

我們給上面的代碼添加一行,如下所示:

public class PassByValue {
    public static void main(String[] args) {
        char[] name = {'磊', '哥'};
        System.out.println("調(diào)用方法前:" + new String(name));
        paramTest(name);
        System.out.println("調(diào)用方法后:" + new String(name));
    }
    private static void paramTest(char[] n) {
        n = new char[2]; // 添加此行代碼
        n[1] = '神';
        System.out.println("方法中修改為:" + new String(n));
    }
}

程序的執(zhí)行結(jié)果為:

調(diào)用方法前:磊哥

方法中修改為:神

調(diào)用方法后:磊哥

從上述結(jié)果可以看出,當(dāng)我們在 paramTest 方法中添加 new char[]之后,“引用傳遞”就突然變值傳遞了?為什么?

這是因?yàn)椋?strong>在 Java 語言中本質(zhì)上只有值傳遞,也就說 Java 的傳參只會傳遞它的副本,并不會傳遞參數(shù)本身。

前面那個帶引號的“引用傳遞”其實(shí)只是傳遞了它的引用副本,如下圖所示:

Java是值傳遞還是引用傳遞?有圖為證

PS:《Java虛擬機(jī)規(guī)范》中對 Java 堆的描述是:“所有的對象實(shí)例以及數(shù)組都應(yīng)當(dāng)在堆上分配”。

所以我們在調(diào)用 new char[] 之后,可以看出 n 對象有了新地址,而原內(nèi)容并未被修改,如果按照引用傳遞的思路來看的話,不管執(zhí)行任何方式的修改都會改變原內(nèi)容,因此我們可以更加確認(rèn) Java 語言中只有值傳遞,如下圖所示:

Java是值傳遞還是引用傳遞?有圖為證

總結(jié)

通過本文的內(nèi)容,我們可以得出:在 Java 語言中只有值傳遞,方法傳參時只會傳遞副本信息而非原內(nèi)容。我們還知道了基礎(chǔ)數(shù)據(jù)類型會直接生成到棧上,而對象或數(shù)組則會在棧和堆上都生成信息,并將棧上生成的引用,直接指向堆中生成的數(shù)據(jù),如下圖所示:

Java是值傳遞還是引用傳遞?上圖為證

以上就是W3Cschool編程獅關(guān)于Java是值傳遞還是引用傳遞?有圖為證的相關(guān)介紹了,希望對大家有所幫助。

0 人點(diǎn)贊