Java只有值傳遞
阿新 • • 發佈:2021-01-03
簡書同步 Java只有值傳遞
為什麼寫這篇文章
網路上說這個問題的文章非常多,但是真的能說明白的目前我沒有找到。而且很大一部分是錯誤的,所以這裡試著用通俗易懂的方式把這個話題說清楚,也希望大家對這個問題有個更好的理解
基本資料型別 值傳遞
基本資料型別使用值傳遞
這個基本沒有什麼問題。這個沒有誤解,先撇開基本型別值傳遞不談了。
思考
- 如果引數是引用型別,就是引用傳遞?這句話對嗎?
很多人以為傳遞的是物件的引用所以是引用傳遞,理解的誤區就在這
先丟擲結論 Java 沒有引用傳遞 官方說的 而且絕對正確
- Java 沒有引用傳遞
- Java 傳遞物件 確實是傳遞物件的引用。更確切的說是傳遞的物件引用的副本。
- Java傳遞的是引用的副本。而不是引用本身,所以引用沒有傳遞,所以這個不叫引用傳遞。
什麼才叫引用傳遞? 為什麼那麼多人有誤解
如果把引用也看做一個物件。那麼引用傳遞的意思就是把這個引用物件傳遞給這個方法。而不是傳遞這個引用的副本。Java裡傳遞的就是引用的副本,雖然這個副本也是引用。但是在Java中只能說是傳遞物件的引用,而不能叫引用傳遞。
首先這不是一個文字遊戲,咬文嚼字,也沒有意義。
因為有很多語言有引用傳遞,所以引用傳遞的概念是明確的。而且具體內容也是明確的。跟Java的物件傳遞是有本質區別的。
如何理解什麼叫傳遞引用本身呢?
這裡因為Java沒有引用傳遞,所以先用C++ 簡單解釋一下,程式碼仿照Java程式碼 class TV{ public String name; } class User{ public TV getTV(&TV tv){//C++可以這樣來傳遞引用 而不是物件引用的副本 &TV 代表傳入一個引用 //這裡傳遞一個TV 的引用 這個引用跟Java的引用是有區別的 tv = new TV("xiaomi");//我修改為了小米牌的 //如果在Java裡 這句話不會影響傳入的tv 但是 C++ 會影響 } } public static void main(String[] args) { TV tv = new TV("shape"); // 這個是夏普牌的電視 User son = new User(); // User father = new User(); son.getTV(tv); System.out.println(" TV is " + tv); // 我這是用Java寫的虛擬碼 如果在Java裡 這裡列印的是shape 但是C++ 這個地方已經指向了xiaomi // 這就是引用傳遞和非引用傳遞的本質區別了 }
引用傳遞是真的把這個引用傳遞給了這個方法,這個方法修改引用的值,會改變傳入的引用本身指向的值,這個傳遞叫做引用傳遞而Java只是傳遞引用的副本,所有這個不能叫引用傳遞。在Java裡只能叫傳遞了物件的引用。而本質確是傳遞了物件的引用的副本。
最好理解的引用傳遞
這裡用遙控器和電視機來簡單理解什麼是引用傳遞。
一臺電視機
一個遙控器
這個遙控器可以操作電視機。
假設我們有一個電視機的物件
TV tv = new TV(); //建立了一個新的電視機物件。 這裡的tv 是指向這個物件的 遙控器。 是一個指向電視機的引用。 這裡應該很好理解。我只是把這個引用換了 一個概念而已 tv.changeChannel("CCTV5");// 通過遙控器調整頻道。 tv.changeChannel("CCTV1");// 通過遙控器調整頻道。 tv.changeChannel("CCTV9");// 通過遙控器調整頻道。 這裡也很好理解 現在我爸想看湖南衛視,他跟我說把遙控器給他。 Father{ public TV getTV(TV tv); } Father father = new Father(); TV fatherTV = father.getTV(tv); fatherTV.changeChannel("HNTV"); 問題就在這裡了。 Java中傳遞的物件的引用對吧?其實這句話是錯的。Java中傳遞的是物件的引用的副本。 只是我們為了方便。直接說傳遞物件的引用,因為引用的副本也是物件的引用 這裡還沒有很繞,還是好理解的。 那麼 在 電視機 和遙控器 的概念裡。這個傳遞是怎麼傳遞的呢 有兩種方式。 第一種,我有一個遙控器直接給了 Father; 第二種,我新買了一個能遙控這臺電視的遙控器給了Father。 第一種 是 我把遙控器傳遞給了 Father 第二種 是 我拿了另一個遙控器給了 Father 這就是本質的區別。 第一種就是 引用傳遞。是遙控器真的傳遞給了Father。遙控器代表引用,所以是引用傳遞給了Father 而第二種 就是值傳遞。只是給了他一個新的遙控器。而我的遙控器沒有傳遞。所以不能叫 遙控器傳遞 也就不能叫 引用傳遞。 用文字理解就是 這個概念。
總結
- Java 沒有引用傳遞
- Java傳遞的引用的副本。而不是引用本身,所以引用沒有傳遞,所以這個不叫引用傳遞。
- Java傳遞的是物件的引用,更確切的說是傳遞的物件引用的副本。
理解
引用傳遞就是引用本身,傳遞給方法,所以叫引用傳遞。而Java傳遞的是引用物件的副本。所以Java的引用型別的引數傳遞不能叫引用傳遞。
Java引用物件的傳遞,傳遞的是引用型別的副本,或者說是引用型別的值,所以說Java是值傳遞,而不是引用傳遞。
其實這個概念確實非常繞,但是理解了之後對Java為什麼不能實現 swap(Object &a, Object &b)還是有幫助的
就是因為Java沒有引用傳遞,所以才無法實現swap 函式。