写在开头
Java是值传递还是引用传递?这个问题几乎100%的出现在了各大主流Java面试题中,知识点很小,但很考验面试者对于Java运行的理解,今晚趁着生产投产的空子,过来小聊一下。
实参与形参
所谓的值传递or引用传递是指方法在调用的过程中实参传递的两种变现形式,那么好,想搞清楚这个问题的前提是,先搞清楚实参与形参
实际参数(实参,英文:Arguments):用于传递给函数/方法的参数,必须有确定的值!
形式参数(形参,英文:Parameters):用于定义函数/方法,接收实参,不需要有确定的值。
int a = 10;
//这里传入的a为实参,有实际确认的值:10
sum(a);
//这里方法定义中的参数为形参,无需确认值,在方法调用中用来接收实际参数
void sum(int p){
System.out.println(p+10);
}
值传递与引用传递
为了充分调用大家的思考,这里先不给结论,先上几段代码示例,通过分析代码,最终得出结果,这是个人最喜欢的总结方式!
【代码示例1-基本数据类型的参数传递】
public class Test {
public static void main(String[] args) {
int a = 1;
int b = 2;
swap(a, b);
System.out.println("实参a = " + a);
System.out.println("实参b = " + b);
}
public static void swap(int p1, int p2) {
int temp = p1;
p1 = p2;
p2 = temp;
System.out.println("p1 = " + p1);
System.out.println("p2 = " + p2);
}
}
输出:
p1 = 2
p2 = 1
实参a = 1
实参b = 2
分析:
在swap()方法中,我们将p1,p2参数的值进行了互换,但这并没有影响实参a,b的值,因为在传入方法时,只是对a,b的值做了拷贝,拷贝之后,他们之间的关系互相独立,这就是值传递!
【代码示例2-引用类型的参数传递】
public class Test {
public static void main(String[] args) {
int[] arr = {
1, 2, 3};
System.out.println(arr[0]);
change(arr);
System.out.println(arr[0]);
}
public static void change(int[] array) {
array[0] = 0;
}
}
输出:
1
0
分析:
初始化一个引用类型的数组arr作为实参,在change方法中我们将数组的0位数值进行了重新赋值,将实参传入方法执行后,我们可以看到实参的第0位数值已经被成功修改为0,看上去是不是像引用传递?
实则不然!请看下面的示意图:
我们知道实参arr是一个对象的引用,而在调用change()时将实参传进来,其实是拷贝了一份实参的引用地址过来,而这个时候实现与形参,他们会执行对象数据对象的同一个地址,导致我们在修改形参的数组值时,实参的0位数值也发生了改变,毕竟他们指向的是同一对象!
【代码示例3-String的参数传递】
public class Test{
public static void main(String[] args) {
Test test= new Test();
// String类
String s = "hello";
test.pass(s);
System.out.println("s = " + s);
}
public void pass(String str) {
str = "world";
System.out.println("str = "+ str);
}
}
输出:
str = world
s = hello
分析:
看到这个结果时,是不是有点出乎意料,在示例2中我们得知,引用类型时的参数传递,拷贝的是引用地址的值,实参会随着形参的改变而改变,但这一段代码的输出显然不符合预期,这是为什么?
看过俺之前写的文章的朋友应该是清楚的,虽然String也是引用类型,但它是不可变类,一旦对赋值完成,就改变不聊了,这也就意味着,上述代码中的两次字符串赋值,分别操作的是两个String对象,两者毫无关联!
总结
看完了如上的例子,大家对Java的参数传递有没有更深的了解了呢,通过上面的几个例子我们可以得出这样的总结:
Java中将实参传递给方法(或函数)的方式是值传递
如果参数是基本类型的话,很简单,传递的就是基本类型的字面量值的拷贝,会创建副本。
如果参数是引用类型,传递的就是实参所引用的对象在堆中地址值的拷贝,同样也会创建副本。