前几天在一套面试题中遇到了一道关于Java的参数传递方式的问题,当时没有回答正确。后来在查阅资料后对这个问题有了详细的了解,在此记录一下。
在给类编程语言中,参数传递的方式有两种。一种是值传递,另一种是引用传递。首先来明确一下两种传递方式具体是什么意思。
值传递是指在方法调用时,传递给方法的参数,实际上是将原来的变量复制一份之后,将复制品传递给了被调用的方法。因为被调用方法得到的是原变量的一个拷贝,所以无论这个方法怎么修改参数,都不会改变原变量的值。
而引用传递是指在方法调用时,是将原变量的引用,也就是变量在内存中的地址空间,传给了被调用方法。因此,被调用方法中的参数和原变量指向的是同一段内存空间。当被调用方法修改参数所指的内存空间内部的值时,原变量的值也会改变。
对于Java的参数传递方式,有两种说法,有人说Java只有值传递,有人说Java既有值传递,又有引用传递。其实Java的传递方式到底叫什么并不重要,重要的是了解在不同的条件下,Java进行参数传递时,内存中到底发生了什么样的变化。
参数传递时内存的改变情况与所传递参数的类型有关。我们知道Java有两类变量类型:基本类型,比如int,float,double等;引用类型:比如数组,String,ArrayList等。这两类数据类型,在内存分配上的区别在于:基本类型的变量名所指向的内存中,存储的就是变量本身的值;而引用类型的变量名所指向的内存中,存储的是一个内存地址,这个内存地址内存储的才是引用类型本身。
对于基本类型,Java就是讲原类型复制一份,将拷贝传递给被调用函数。举例如下:
public class MyTest {
public static void change(int b) {
b = 6;
}
public static void main(String[] args) {
int a = 5;
change(a);
System.out.println("a=" + a);// 输出a=5
}
}
可见,我们在change()方法中对b进行了重新复制,但并没有影响a的值。两者在内存中的分配方式如下:
对于引用类类型,可以确定的是Java是将其在内存中的地址传递给了被调用方法。但是,当方法对参数进行改变时,是否会修改原参数则需要分情况讨论。
情况一:直接修改参数内部的值,这时,修改会改变原变量的值:
public class MyTest {
public static void change(int[] b) {
for(int i = 0; i<b.length; i++){
b[i] = b[i] * 2;
}
}
public static void main(String[] args) {
int[] a = new int[]{1,2,3};
change(a);
for(int i = 0; i<a.length; i++){
System.out.print(a[i] + " ");
}
// 输出结果为2 4 6
}
}
内存中的情况为:
public class MyTest {
public static void change(int[] b) {
b = new int[]{2,4,6};
}
public static void main(String[] args) {
int[] a = new int[]{1,2,3};
change(a);
for(int i = 0; i<a.length; i++){
System.out.print(a[i] + " ");
}
// 输出结果为1 2 3
}
}
内存中的情况为:
public class MyTest {
public static void change(String b) {
b += "cde";
}
public static void main(String[] args) {
String s = "abc";
change(s);
System.out.println(s);
// 输出结果为abc
}
}
public class MyTest {
public static void change(String b) {
b = new String("cde");
}
public static void main(String[] args) {
String s = "abc";
change(s);
System.out.println(s);
// 输出结果为abc
}
}