前言
提示:首先我们需要了解下两个概念:
- 形参&实参
- 值传递&引用传递
形参&实参
方法的定义可能会用到参数,参数在方法中分为:
- 实参(实际参数):用于传递给函数/方法的参数,必须有值
- 形参(形式参数):用于定义函数/方法,用于接受实参,不需要有确定的值
String hello = "Hello!";
// hello 为实参
sayHello(hello);
// str 为形参
void sayHello(String str) {
System.out.println(str);
}
值传递&引用传递
- 值传递:方法接受的是实参值的拷贝,会创建副本
- 引用传递:方法接受的是实参所引用的对象所在堆中的地址,不会创建副本,对形参的修改将影响到实参
为什么说Java中只有值传递?
案例一:传递基本参数
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拷贝过来的,相当于他们俩的副本,所以不会影响原件本身
案例二:传递引用类型参数一
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
这里可能会认为Java对于引用类型是引用传递,其实也不是的!
这里是因为array拷贝了arr(实参)的地址,它和arr指向的是同一个对象,这就说明了为什么方法内部对形参的修改也会影响到实参
案例三:传递引用类型二
public class Person {
private String name;
// 省略构造函数、Getter&Setter方法
}
public static void main(String[] args) {
Person xiaoZhang = new Person("小张");
Person xiaoLi = new Person("小李");
swap(xiaoZhang, xiaoLi);
System.out.println("xiaoZhang:" + xiaoZhang.getName());
System.out.println("xiaoLi:" + xiaoLi.getName());
}
public static void swap(Person person1, Person person2) {
Person temp = person1;
person1 = person2;
person2 = temp;
System.out.println("person1:" + person1.getName());
System.out.println("person2:" + person2.getName());
}
输出:
person1:小李
person2:小张
xiaoZhang:小张
xiaoLi:小李
解析:
这里就能很明显的否认Java不是引用传递!
swap方法中的参数person1,person2只是拷贝了原件在堆中的地址,因此方法中互换的只是他们两个的地址罢了,并不会影响到实参xiaoZhang,和xiaoLi
总结
Java 中将实参传递给方法(或函数)的方式是 值传递 :
- 如果参数是基本类型的话,传递的就是基本类型的字面量值的拷贝,会创建副本。
- 如果参数是引用类型,传递的就是实参所引用的对象在堆中地址值的拷贝,同样也会创建副本。