很多人不知道Java属于值传递还是引用传递,大多数可能认为Java传基本类型时是值传递,传对象时是引用传递,那么事实真是如此吗?
简介
??首先回顾一下在程序设计语言中有关将参数传递给方法(或函数)的一些专业术语:按值调用 (call by value)表示方法接收的是调用者提供的值,而按引用调用(call by reference)表示方法接收的是调用者提供的变量地址(即形参和实参共用同一存储空间)。一个方法不能修改值传递调用所对应的实参值,而可以修改引用传递所对应的实参值。 它用来描述各种程序设计语言(不只是Java)中方法参数传递方式。
传参原理
??那么从上面的定义来看,在传递参数到方法执行过程中,值传递不能通过更改形参,而改变实参,引用传递可以通过更改形参而更改实参,说白了就是,值传递中形参和实参是复制的关系;而在引用传递中,形参和实参共用内存,那么,在Java中是什么情况呢?请看示例:
Java中基础类型传递
public static void main(String[] args) {
int n1 = 10;
int n2 = 20;
swap(n1, n2);
System.out.println("n1 = " + n1);
System.out.println("n2 = " + n2);
}
// 交换方法
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
n1 = 10
n2 = 20
解析: ??在 swap(int a, int b) 方法中,a、b 的值进行交换,并没有影响到 n1、n2。因为,a、b 中的值,只是从 n1、n2 的复制过来的。相当于 n1、n2 的副本,副本的内容无论怎么修改,都不会影响到原件本身。所以Java中基础类型传递是值传递,这是没有争议的。
Java中对象传递
public static void main(String[] args) {
Student s1 = new Student("张三");
Student s2 = new Student("李四");
swap(s1, s2);
System.out.println("s1: " + s1.getName());
System.out.println("s2: " + s2.getName());
}
// 交换方法
public static void swap(Student a, Student b) {
Student temp = a;
a = b;
b = temp;
System.out.println("a: " + a.getName());
System.out.println("b: " + b.getName());
}
结果:
a: 李四
b: 张三
s1: 张三
s2: 李四
解析:
??如图所示,方法并没有改变存储在变量 s1 和 s2 中的对象引用指向。 而只是把拷贝副本的指向进行了交换,原实参s1、s2并未改变。此时可能有人会说,“我可以通过 a 修改 student 1 的值,s1指向的对象不就变了,不就是传递引用,改变了实参值吗?” 这是Java对象是引用传递支持者最大的举证,但有一个很大的误区。即我们传的是对象引用,而我们改正的是什么,是对象,这本就是两个东西,你并没有修改你传的实参值(对象引用)的指向,而在C++中引用参数标有 & 符号。 可以轻松地实现void swap(double& a, double& b)方法或 void swap(Student& a, Student& b) 方法实现修改它们的实参值(引用参数)的目的,既通过交换形参,而交换实参,因为它们本质用的是同一块内存。
总结
??在Java中,无论是基础类型传递还是对象传递,总是按值传递的。
参考
《Java 核心技术卷 I》
|