在Java中字符串是通過引用傳遞的?


這是一個經典的java問題。在stackoverflow上,許多類似的問題已經被問過了,但是許多回答是錯誤的或不完整的。

 

如果你不想深入思考的話,這個問題很簡單。如果你想明白的更徹底,那么問題可能容易讓人迷惑不清。

 

1. 一段有趣但令人困惑的代碼


 

public static void main(String[] args) {
  String x = new String("ab");
  change(x);
  System.out.println(x);
}
public static void change(String x) {
    x = "cd";
}

 

輸出 "ab"

 

C++ 代碼如下:

void change(string &x) {
    x = "cd";
}
int main(){    
    string x = "ab";
    change(x);    
    cout << x << endl;
}

 

輸出 "cd"

 

2.經常讓人捉摸不透的問題


 

x 存儲了堆中"ab"字符串的引用。因此,當x作為參數傳遞到change()方法的時候,它仍然堆中的"ab",如下所示:

 

因為java是按值傳遞的,x的值是"ab"的引用。當方法change()被調用的時候,它創建了一個新的字符串對象"cd" ,然后x就指向了"cd" ,如圖所示:

這似乎是一個非常合理的解釋。他們很清楚,java是按值傳遞的,但是這里出了什么問題?

 

3.這段代碼到底做了什么?


 

上面的解釋有幾處錯誤。為了更容易理解,我們最好簡單的過一下整個流程。

當字符串"cd" 被創建時,java會分配儲存字符串所需要的內存量。然后,對象被分配給了變量x,實際上是將對象的引用分配給了變量x。這個引用是對象儲存的內存地址。

 

變量x包含了一個指向字符串對象的引用,x並不是字符串對象本身。它是一個儲存了字符串對象'ab'引用的變量。

 

java是按值傳遞的。當x被傳遞給change()方法時,實際上是x的值(一個引用)的一個副本。方法change被調用后,會創建另一個對象"cd",它有着一個不同的引用。方法內的局部變量x的值變成了"cd"的引用。這里改變的是方法內的局部變量的引用值,而不是改不了原先引用的字符串"ab"。

看圖:

 

 

4.錯誤的解釋:


 

從第一個代碼片段引發的問題與字符串不可變性沒有任何關系。即使String被StringBuilder替換,結果仍然是一樣的。關鍵點是變量存儲的是對象引用,而不是對象本身!

 

5.解決這個問題


 

如果我們真的需要去改變對象的值,首先,對象應該是可變的,比如StringBuilder。其次,我們需要確保沒有創建新對象並將其分配給參數變量,因為Java只是按值傳遞。

public static void main(String[] args) {
  StringBuilder x = new StringBuilder("ab");
  change(x);
  System.out.println(x);
}
public static void change(StringBuilder x) {
  x.delete(0, 2).append("cd");
}

 

作者:在下賊溜
鏈接:https://www.jianshu.com/p/df903886b5a9
來源:簡書

關注我們"java工會",獲取更多java技術干貨,等你來撩!


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM