形参&实参

方法的定义可能会用到 参数(有参的方法),参数在程序语言中分为:

  • 实参(实际参数) :用于传递给函数/方法的参数,必须有确定的值。
  • 形参(形式参数) :用于定义函数/方法,接收实参,不需要有确定的值。

    1. String hello = "Hello!";
    2. // hello 为实参
    3. sayHello(hello);
    4. // str 为形参
    5. void sayHello(String str) {
    6. System.out.println(str);
    7. }

    值传递&引用传递

    程序设计语言将实参传递给方法(或函数)的方式分为两种:

  • 值传递 :方法接收的是实参值的拷贝,会创建副本。

  • 引用传递 :方法接收的直接是实参所引用的对象在堆中的地址,不会创建副本,对形参的修改将影响到实参。

很多程序设计语言(比如 C++、 Pascal )提供了两种参数传递的方式,不过,在 Java 中只有值传递。

案例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

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

案例2:传递引用类型参数1

  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

解析:
image.png
看了这个案例很多人肯定觉得 Java 对引用类型的参数采用的是引用传递。
实际上,并不是的,这里传递的还是值,不过,这个值是实参的地址罢了!
也就是说 change 方法的参数拷贝的是 arr (实参)的地址,因此,它和 arr 指向的是同一个数组对象。这也就说明了为什么方法内部对形参的修改会影响到实参。
为了更强有力地反驳 Java 对引用类型的参数采用的不是引用传递,我们再来看下面这个案例!

案例3 :传递引用类型参数2

  1. public class Person {
  2. private String name;
  3. // 省略构造函数、Getter&Setter方法
  4. }
  5. public static void main(String[] args) {
  6. Person xiaoZhang = new Person("小张");
  7. Person xiaoLi = new Person("小李");
  8. swap(xiaoZhang, xiaoLi);
  9. System.out.println("xiaoZhang:" + xiaoZhang.getName());
  10. System.out.println("xiaoLi:" + xiaoLi.getName());
  11. }
  12. public static void swap(Person person1, Person person2) {
  13. Person temp = person1;
  14. person1 = person2;
  15. person2 = temp;
  16. System.out.println("person1:" + person1.getName());
  17. System.out.println("person2:" + person2.getName());
  18. }

输出:

  1. person1:小李
  2. person2:小张
  3. xiaoZhang:小张
  4. xiaoLi:小李

解析:
怎么回事???两个引用类型的形参互换并没有影响实参啊!
swap 方法的参数 person1 和 person2 只是拷贝的实参 xiaoZhang 和 xiaoLi 的地址。因此, person1 和 person2 的互换只是拷贝的两个地址的互换罢了,并不会影响到实参 xiaoZhang 和 xiaoLi 。
image.png

总结

Java 中将实参传递给方法(或函数)的方式是 值传递

  • 如果参数是基本类型的话,很简单,传递的就是基本类型的字面量值的拷贝,会创建副本。
  • 如果参数是引用类型,传递的就是实参所引用的对象在堆中地址值的拷贝,同样也会创建副本。

——- 著作权归Guide哥所有。 链接: https://javaguide.cn/java/basis/why-there-only-value-passing-in-java/#%E6%A1%88%E4%BE%8B3-%E4%BC%A0%E9%80%92%E5%BC%95%E7%94%A8%E7%B1%BB%E5%9E%8B%E5%8F%82%E6%95%B02