首先回顾一下在程序设计语言中有关将参数传递给方法(或函数)的一些专业术语。按值调用(call by value)表示方法接收的是调用者提供的值,而按引用调用(call by reference)表示方法接收的是调用者提供的变量地址。一个方法可以修改传递引用所对应的变量值,而不能修改传递值调用所对应的变量值。 它用来描述各种程序设计语言(不只是 Java)中方法参数传递方式。

    Java 程序设计语言总是采用按值调用。也就是说,方法得到的是所有参数值的一个拷贝,也就是说,方法不能修改传递给它的任何参数变量的内容。

    下面通过 3 个例子来给大家说明**

    example 1

    1. public static void main(String[] args) {
    2. int num1 = 10;
    3. int num2 = 20;
    4. swap(num1, num2);
    5. System.out.println("num1 = " + num1);
    6. System.out.println("num2 = " + num2);
    7. }
    8. public static void swap(int a, int b) {
    9. int temp = a;
    10. a = b;
    11. b = temp;
    12. System.out.println("a = " + a);
    13. System.out.println("b = " + b);
    14. }

    结果:

    1. a = 20
    2. b = 10
    3. num1 = 10
    4. num2 = 20

    解析:
    为什么 Java 中只有值传递? - 图1
    在 swap 方法中,a、b 的值进行交换,并不会影响到 num1、num2。因为,a、b 中的值,只是从 num1、num2 的复制过来的。也就是说,a、b 相当于 num1、num2 的副本,副本的内容无论怎么修改,都不会影响到原件本身。
    通过上面例子,我们已经知道了一个方法不能修改一个基本数据类型的参数,而对象引用作为参数就不一样,请看 example2.

    example 2

    1. public static void main(String[] args) {
    2. int[] arr = { 1, 2, 3, 4, 5 };
    3. System.out.println(arr[0]);
    4. change(arr);
    5. System.out.println(arr[0]);
    6. }
    7. public static void change(int[] array) {
    8. // 将数组的第一个元素变为0
    9. array[0] = 0;
    10. }

    结果:

    1. 1
    2. 0

    解析:
    为什么 Java 中只有值传递? - 图2
    array 被初始化 arr 的拷贝也就是一个对象的引用,也就是说 array 和 arr 指向的是同一个数组对象。 因此,外部对引用对象的改变会反映到所对应的对象上。

    通过 example2 我们已经看到,实现一个改变对象参数状态的方法并不是一件难事。理由很简单,方法得到的是对象引用的拷贝,对象引用及其他的拷贝同时引用同一个对象。

    很多程序设计语言(特别是,C++和 Pascal)提供了两种参数传递的方式:值调用和引用调用。有些程序员(甚至本书的作者)认为 Java 程序设计语言对对象采用的是引用调用,实际上,这种理解是不对的。由于这种误解具有一定的普遍性,所以下面给出一个反例来详细地阐述一下这个问题。**

    example 3

    1. public class Test {
    2. public static void main(String[] args) {
    3. // TODO Auto-generated method stub
    4. Student s1 = new Student("小张");
    5. Student s2 = new Student("小李");
    6. Test.swap(s1, s2);
    7. System.out.println("s1:" + s1.getName());
    8. System.out.println("s2:" + s2.getName());
    9. }
    10. public static void swap(Student x, Student y) {
    11. Student temp = x;
    12. x = y;
    13. y = temp;
    14. System.out.println("x:" + x.getName());
    15. System.out.println("y:" + y.getName());
    16. }
    17. }

    结果:

    1. x:小李
    2. y:小张
    3. s1:小张
    4. s2:小李

    解析:
    交换之前:
    为什么 Java 中只有值传递? - 图3
    交换之后:
    为什么 Java 中只有值传递? - 图4
    通过上面两张图可以很清晰的看出: 方法并没有改变存储在变量 s1 和 s2 中的对象引用。swap 方法的参数 x 和 y 被初始化为两个对象引用的拷贝,这个方法交换的是这两个拷贝

    总结

    Java 程序设计语言对对象采用的不是引用调用,实际上,对象引用是按 值传递的。
    下面再总结一下 Java 中方法参数的使用情况:

    • 一个方法不能修改一个基本数据类型的参数(即数值型或布尔型)。
    • 一个方法可以改变一个对象参数的状态。
    • 一个方法不能让对象参数引用一个新的对象。

    参考:
    《Java 核心技术卷 Ⅰ》基础知识第十版第四章 4.5 小节