1. 程式人生 > 實用技巧 >Java中只有值傳遞!!!

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 中方法引數的使用情況:

  • 一個方法不能修改一個基本資料型別的引數(即數值型或布林型)。
  • 一個方法可以改變一個物件引數的狀態。
  • 一個方法不能讓物件引數引用一個新的物件。