Java中只有值傳遞!!!
首先回顧一下在程式設計語言中有關將引數傳遞給方法(或函式)的一些專業術語。按值呼叫(call by value)表示方法接收的是呼叫者提供的值,而按引用呼叫(call by reference)表示方法接收的是呼叫者提供的變數地址。一個方法可以修改傳遞引用所對應的變數值,而不能修改傳遞值呼叫所對應的變數值。它用來描述各種程式設計語言(不只是 Java)中方法引數傳遞方式。
Java 程式設計語言總是採用按值呼叫。也就是說,方法得到的是所有引數值的一個拷貝,也就是說,方法不能修改傳遞給它的任何引數變數的內容。
下面通過 3 個例子來給大家說明
example 1
public static void main(String[] args) {
int num1 = 10;
int num2 = 20;
swap(num1, num2);
System.out.println("num1 = " + num1);
System.out.println("num2 = " + num2);
}
public static void swap(int a, int b) {
int temp = a;
a = b;
b = temp;
System.out.println("a = " + a);
System.out.println("b = " + b);
}
結果:
a = 20
b = 10
num1 = 10
num2 = 20
解析:
在 swap 方法中,a、b 的值進行交換,並不會影響到 num1、num2。因為,a、b 中的值,只是從 num1、num2 的複製過來的。也就是說,a、b 相當於 num1、num2 的副本,副本的內容無論怎麼修改,都不會影響到原件本身。
通過上面例子,我們已經知道了一個方法不能修改一個基本資料型別的引數,而物件引用作為引數就不一樣,請看 example2.
example 2
public static void main(String[] args) {
int[] arr = { 1, 2, 3, 4, 5 };
System.out.println(arr[0]);
change(arr);
System.out.println(arr[0]);
}
public static void change(int[] array) {
// 將陣列的第一個元素變為0
array[0] = 0;
}
結果:
1
0
解析:
array 被初始化 arr 的拷貝也就是一個物件的引用,也就是說 array 和 arr 指向的是同一個陣列物件。 因此,外部對引用物件的改變會反映到所對應的物件上。
通過 example2 我們已經看到,實現一個改變物件引數狀態的方法並不是一件難事。理由很簡單,方法得到的是物件引用的拷貝,物件引用及其他的拷貝同時引用同一個物件。
很多程式設計語言(特別是,C++和 Pascal)提供了兩種引數傳遞的方式:值呼叫和引用呼叫。有些程式設計師(甚至本書的作者)認為 Java 程式設計語言對物件採用的是引用呼叫,實際上,這種理解是不對的。由於這種誤解具有一定的普遍性,所以下面給出一個反例來詳細地闡述一下這個問題。
example 3
public class Test {
public static void main(String[] args) {
// TODO Auto-generated method stub
Student s1 = new Student("小張");
Student s2 = new Student("小李");
Test.swap(s1, s2);
System.out.println("s1:" + s1.getName());
System.out.println("s2:" + s2.getName());
}
public static void swap(Student x, Student y) {
Student temp = x;
x = y;
y = temp;
System.out.println("x:" + x.getName());
System.out.println("y:" + y.getName());
}
}
結果:
x:小李
y:小張
s1:小張
s2:小李
解析:
交換之前:
交換之後:
通過上面兩張圖可以很清晰的看出:方法並沒有改變儲存在變數 s1 和 s2 中的物件引用。swap 方法的引數 x 和 y 被初始化為兩個物件引用的拷貝,這個方法交換的是這兩個拷貝
總結
Java 程式設計語言對物件採用的不是引用呼叫,實際上,物件引用是按 值傳遞的。
下面再總結一下 Java 中方法引數的使用情況:
- 一個方法不能修改一個基本資料型別的引數(即數值型或布林型)。
- 一個方法可以改變一個物件引數的狀態。
- 一個方法不能讓物件引數引用一個新的物件。