Java中的参数传递很多人都弄不明白是怎么回事,有人说是值传递,有人说是引用传递,或者有人说既是值传递也是引用传递。那么今天就来看看,Java中的参数传递,到底是传值,还是传的引用。
先说结论:
Java中方法参数传递方式是按值传递:
- 如果参数是基本类型,传递的是基本类型的字面量值的拷贝。
- 如果参数是引用类型,传递的是该参量所引用的对象在堆中地址值的拷贝
对于第一个结论理解很简单,举个例子来说明:
public class Main {
public static void main(String[] args) {
Person p = new Person();
int n = 15;
p.setAge(15);
System.out.println(p.getAge());//打印出来为15
n = 20;
System.out.println(p.getAge());//打印出来为15
}
}
class Person {
private int age;
public int getAge() {
return this.age;
}
public void setAge(int age) {
this.age = age;
}
}
从结果可知,修改外部的局部变量n,不影响实例p的age字段,原因是setAge()方法获得的参数,复制了n的值,因此,p.age和局部变量n互不影响。
结论:基本类型参数的传递,是调用方值的复制。双方各自的后续修改,互不影响。
这一点很好理解,没有什么问题。那么对于第二个结论我们来举两个例子来说明。
例子1:
public class Main {
public static void main(String[] args) {
Person p = new Person();
int n = 15;
p.setAge(15);
System.out.println(p.getAge());//打印出来为15
System.out.println(System.identityHashCode(p));//打印出地址为764977973
fun(p);
System.out.println(p.getAge());//打印出来为20
System.out.println(System.identityHashCode(p));//打印出地址为764977973
}
public static void fun(Person person) {
person.setAge(20);
System.out.println(System.identityHashCode(p));//打印出地址为764977973
}
}
class Person {
private int age;
public int getAge() {
return this.age;
}
public void setAge(int age) {
this.age = age;
}
}
例子2:
public class Main {
public static void main(String[] args) {
Person p = new Person();
int n = 15;
p.setAge(15);
System.out.println(p.getAge());//打印出来为15
System.out.println(System.identityHashCode(p));//打印出地址为764977973
fun(p);
System.out.println(p.getAge());//打印出来为15
System.out.println(System.identityHashCode(p));//打印出地址为764977973
}
public static void fun(Person person) {
person = new Person();
person.setAge(20);
System.out.println(System.identityHashCode(person));//打印出地址为381259350
}
}
class Person {
private int age;
public int getAge() {
return this.age;
}
public void setAge(int age) {
this.age = age;
}
}
先说下程序中对于值传递和引用传递的定义为:
- 值传递:指在调用函数时将实际参数复制一份传递到函数中,这样在函数中如果对参数进行修改,将不会影响到实际参数。
- 引用传递:指在调用函数时将实际参数的地址传递到函数中,那么在函数中对参数所进行的修改,将影响到实际参数。
两者的区别为:
- 值传递会创建副本,函数中无法改变原始对象
- 引用传递不创建副本,函数中可以改变对象
那么有人就会说,第一个例子就说明了会改变对象,不就证明了Java是引用传递嘛。
事实真的是这样的吗?其实通过第二个例子可以很好的说明,java中的参数传递是值传递而不是引用传递。
第二个例子中 fun 方法里new了一个新的对象,如果是引用传递,那么原始对象指向的地址应该是新对象的地址,而结果却是原始对象的地址并没有改变。
那么第一例子为什么会改变原始对象呢?
因为在第一个例子中传递的是对地址的拷贝。
对于引用类型,传递的是该参量所引用的对象在堆中地址值的拷贝