对于一个函数,我们可以传递引用类型或是数值类型给这个函数。然而这两种类型的区别非常大,但却又常常被新手误解。因此,理解他们的区别与用法则显得非常重要。
这里有一张动图可以表示两者之间的区别。简单来说,引用类型指向对象的位置,因此任何通过该引用所做的更改都会影响同一个对象。数值类型则是会复制一份新的数据,所以对复制的数据做任何改动也不会影响原始数据。
传递数值类型
在Java里,如果一个函数接收的参数为原始类型 (primitive types),那这就是属于传递数值类型,则该参数会先被复制一份,再传递给函数。
因此,即使我们对数值类型的拷贝做任何修改,也不会影响原始数据。
public static void plusFive(int a) {
a = a + 5;
}
public static void main(String[] args) {
int a = 6;
plusFive(a);
System.out.println(a); // Output: 6
}
从代码中可以看到,我们首先在 main 函数中声明了一个整型变量 a,分配数值 6 给它。
接着我们将该变量 a 传递给函数 plusFive(),该函数会往它的参数变量上再加 5。
接着,我们尝试在 main 函数中打印出变量 a 的值,我们会得到数值 6,而不是 11。这是因为我们在函数 plusFive() 中所修改的数据只是原始数据的拷贝而已。
传递引用类型
引用类型可以理解为一个指向对象位置的地址。因此,如果我们通过同一个引用类型向对象进行改动,这些改动所影响的都是同一个对象。
public static void main(String[] args) {
Person person = new Person();
Person Tom = person;
Person Kevin = person;
Tom.name = "Tom";
Kevin.name = "Kevin";
System.out.println(Tom.name);
System.out.println(Kevin.name);
// Output:
// Kevin
// Kevin
}
根据上述代码,变量 Tom 与 Kevin 都储存着同一个引用。所以,任何改动所影响的都是同一个对象。因此,我们最后看到的输出是 Kevin 与 Kevin。
在Java里,如果我们向函数传递的是一个对象,则我们实际传递的是该对象的引用。所以,在这里我们仍然只有一个对象而已。因此,如果我们在函数中对该引用的对象做出任何改动,被影响的都是同一个对象。
public static void changeName(Person person) {
person.name = "new name";
}
public void static main(String[] args) {
Person person = new Person();
person.name = "old name";
changeName(person);
System.out.println(person.name); // Output: new name
}
从代码中可以看到,我们在函数 main 中有一个 Person 对象,我们给该对象的 name 赋值 “old name”。
接着我们将该对象传入函数 changeName() 去改变它的名字,改成 “new name”。
接着我们在 main 函数中打印该对象的名字,得到的结果是 “new name”。说明在函数中所改动的对象是同一个对象。
总结
传递数值类型会先对该数值进行拷贝,再将拷贝出来的数值传递给函数。因此任何对于该数值拷贝的改动都不会影响原始数值。
传递引用则是相当于传递一个对象的地址,这里不会发生对象拷贝,即使我们有多个变量储存同一个引用,这些引用指向的都是同一个对象,因此任何通过该引用向对象做出的改动都会影响同一个对象。