1蜈垮、為什么 Java 中只有值傳遞耗跛?
首先回顧一下在程序設(shè)計語言中有關(guān)將參數(shù)傳遞給方法(或函數(shù))的一些專業(yè)術(shù)語裕照。按值調(diào)用(call by value)表示方法接收的是調(diào)用者提供的值,而按引用調(diào)用(call by reference)表示方法接收的是調(diào)用者提供的變量地址课兄。一個方法可以修改傳遞引用所對應(yīng)的變量值牍氛,而不能修改傳遞值調(diào)用所對應(yīng)的變量值。 它用來描述各種程序設(shè)計語言(不只是Java)中方法參數(shù)傳遞方式烟阐。
Java程序設(shè)計語言總是采用按值調(diào)用搬俊。也就是說,方法得到的是所有參數(shù)值的一個拷貝蜒茄,也就是說唉擂,方法不能修改傳遞給它的任何參數(shù)變量的內(nèi)容。
下面通過 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);
}
結(jié)果:
a = 20
b = 10
num1 = 10
num2 = 20
解析:
在swap方法中檀葛,a玩祟、b的值進(jìn)行交換,并不會影響到 num1屿聋、num2空扎。因?yàn)椋琣润讥、b中的值转锈,只是從 num1、num2 的復(fù)制過來的楚殿。也就是說撮慨,a、b相當(dāng)于num1脆粥、num2 的副本砌溺,副本的內(nèi)容無論怎么修改,都不會影響到原件本身变隔。
通過上面例子规伐,我們已經(jīng)知道了一個方法不能修改一個基本數(shù)據(jù)類型的參數(shù),而對象引用作為參數(shù)就不一樣匣缘,請看 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) {
// 將數(shù)組的第一個元素變?yōu)?
array[0] = 0;
}
結(jié)果:
1
0
解析:
array 被初始化 arr 的拷貝也就是一個對象的引用猖闪,也就是說 array 和 arr 指向的時同一個數(shù)組對象。 因此孵户,外部對引用對象的改變會反映到所對應(yīng)的對象上萧朝。
通過 example2 我們已經(jīng)看到岔留,實(shí)現(xiàn)一個改變對象參數(shù)狀態(tài)的方法并不是一件難事夏哭。理由很簡單,方法得到的是對象引用的拷貝献联,對象引用及其他的拷貝同時引用同一個對象竖配。
很多程序設(shè)計語言(特別是何址,C++和Pascal)提供了兩種參數(shù)傳遞的方式:值調(diào)用和引用調(diào)用。有些程序員(甚至本書的作者)認(rèn)為Java程序設(shè)計語言對對象采用的是引用調(diào)用进胯,實(shí)際上用爪,這種理解是不對的。由于這種誤解具有一定的普遍性胁镐,所以下面給出一個反例來詳細(xì)地闡述一下這個問題偎血。
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());
}
}
結(jié)果:
x:小李
y:小張
s1:小張
s2:小李
解析:
交換之前:
交換之后:
通過上面兩張圖可以很清晰的看出: 方法并沒有改變存儲在變量 s1 和 s2 中的對象引用。swap方法的參數(shù)x和y被初始化為兩個對象引用的拷貝盯漂,這個方法交換的是這兩個拷貝
總結(jié)
Java程序設(shè)計語言對對象采用的不是引用調(diào)用颇玷,實(shí)際上,對象引用是按 值傳遞的就缆。
下面再總結(jié)一下Java中方法參數(shù)的使用情況:
一個方法不能修改一個基本數(shù)據(jù)類型的參數(shù)(即數(shù)值型或布爾型)帖渠。
一個方法可以改變一個對象參數(shù)的狀態(tài)。
一個方法不能讓對象參數(shù)引用一個新的對象竭宰。