為什麼Java隻有值傳遞?
形參和實參
形式參數,是在方法定義階段,是定義某個函數時使用的參數,用于接收實參傳入。例f(x,y)中x和y是形參。
實際參數,是在方法調用階段,是主調函數調用有參函數時,實際傳遞的内容。例f(3,7)中3和7是實參。
值傳遞和引用傳遞
值傳遞和引用傳遞不是簡單地通過傳遞内容區分的。如果是值,就是值傳遞;如果是引用,就是引用傳遞。這一了解是不正确的。
值傳遞,是指在調用函數時将實際參數複制一份傳遞給函數形參。此時,在函數中對形參做修改,不影響實際參數。
引用傳遞,是指在調用函數時将實際參數的位址直接傳遞給函數形參。此時,在函數中對參數做修改,将影響實際參數。
根本差別在于值傳遞會建立副本,是以函數中無法改變原始對象;引用傳遞不建立副本,函數中可以改變原始對象。
通過一個經典案例講解Java值傳遞
public class ParamPassing {
private static int intStatic = 222;
private static String stringStatic = "old string";
private static StringBuilder stringBuilderStatic = new StringBuilder("old stringBuilder");
public static void main(String[] args) {
// 方法調用1
method(intStatic);
System.out.println(intStatic);
// 方法調用2
method();
System.out.println(intStatic);
// 方法調用3
method(stringStatic);
System.out.println(stringStatic);
// 方法調用4
method(stringBuilderStatic, stringBuilderStatic);
System.out.println(stringBuilderStatic);
}
// 方法1
public static void method(int intStatic) {
intStatic = 777;
}
// 方法2
public static void method() {
intStatic = 888;
}
// 方法3
public static void method(String stringStatic) {
stringStatic = "new string";
}
// 方法4
public static void method(StringBuilder stringBuilderStatic1, StringBuilder stringBuilderStatic2) {
stringBuilderStatic1.append(".method.first-");
stringBuilderStatic2.append(".method.second-");
// 引用重新指派
stringBuilderStatic1 = new StringBuilder("new stringBuilder");
stringBuilderStatic1.append("new method's append");
}
}
運作結果:
222
888
old string
old stringBuilder.method.first-.method.second-
方法1,參數是局部變量,拷貝的變量值是777,會存入虛拟機棧中的局部變量表的第一個位置。在方法内部,根據作用于就近原則,使用局部變量的參數,操作與實參無關。而方法2,先把本地指派的888壓入虛拟機棧中的操作棧,然後給靜态周遊intStatic指派。
![](https://img.laitimes.com/img/_0nNw4CM6IyYiwiM6ICdiwiIx0DciV2dmADM30zd-cmbw5CRzUCRzUydaVnQuxUeJpmT1UEVNpXRq1EeFpnT3tGVNdXS6xUdO1GTuFzVhVnUyMma1knW2hXbZR3YXJGc5kHT20ESjBjUIF2Lc12bj5SYphXa5VWen5WY35iclN3Ztl2Lc9CX6MHc0RHaiojIsJye.png)
方法3,String是immutable對象,類中沒有提供任何方法用來修改對象。“old string"仍然由實參持有,在方法3中,會重新new一個String對象,并把引用賦給形參。
方法4,直接使用參數引用,可以修改對象;當對引用重新指派後,不再影響實參。
當stringBuilderStatic引用作為實參傳遞給形參stringBuilderStatic1時,此時形參是stringBuilderStatic的一個副本,兩個引用共同指向StringBuilder對象所在的堆記憶體位址,此時對形參的任何修改都會改變對象屬性。當建立新對象并指派給stringBuilderStatic1後,該引用指向了新的記憶體位址,對其修改不會改變原對象的屬性。
位元組碼解釋
0: aload_0 // 引用類型入棧
1: ldc #14 // 将常量值從常量池推到棧頂
// String .method.first-
3: invokevirtual #15 // 調用執行個體方法
// Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
6: pop // 棧頂元素出棧
14: new #17 // 建立執行個體
// class java/lang/StringBuilder
17: dup // 指派棧頂數值,壓入棧頂
18: ldc #18
// String new stringBuilder
20: invokespecial #19 // 執行個體初始化
// Method java/lang/StringBuilder."<init>":(Ljava/lang/String;)V
23: astore_0 // 将棧中ref引用存到局部變量表
24: aload_0 // 加載局部變量到操作數棧
25: ldc #20 // 加載常量到操作數棧
// String new method's append
27: invokevirtual #15 // 調執行個體方法
// Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
30: pop // 操作數棧頂元素出棧
31: return // 方法傳回指令
posted on 2019-07-11 21:37 Awecoder 閱讀(...) 評論(...) 編輯 收藏