前言
这次分享的是一道有迷惑性的代码阅读题,主要考察点在于引用传递和值传递的区别。
代码
求这段代码的输出结果。
class Value{
public int i=15;
}
public class Test{
public static void main(String argv[]){
Test t=new Test( );
t.first( );
}
public void first( ){
int i=5;
Value v=new Value( );
v.i=25;
second(v,i);
System.out.println(v.i);
System.out.println(i);
}
public void second(Value v,int i){
i = 0;
v.i = 20;
Value val = new Value( );
v = val;
System.out.println(v.i+" "+i);
}
}
输出结果
15 0
20
分析
此题的迷惑点在于second()方法里的形参Value v。
- 当first()方法中的实参v传给second()中的形参v时,实际上是执行了Value v = v。实参v的内存地址赋值给了形参v;
- second()方法中先将v对象中的i赋值为20,然后再将val对象的内存地址赋值给v,使v指向val在堆中的内存空间,这时候第一个打印的v.i,其实就是val.i,打印15;
- 第二个打印的i是基本数据类型int,实参和形参之间是值传递,固然此时打印的是i = 0;
- 第三个打印的是v对象的i值,在second()方法中v.i的值被赋值为20. v=val这里是将val的内存地址传递给v,此v非彼v。看下面这段代码:
public void second(Value x,int i){
i = 0;
x.i = 20;
Value val = new Value( );
x = val;
System.out.println(x.i+" "+i);
}
将v换成x,看起来是不是清晰了一点。
在执行x = val之前,x和v都是指向同一个内存空间。如图所示:
在执行完x = val后,x 和val指向同一个内存空间,如图所示:
所以第三次打印的是v.i = 20;
总结
引用传递和值传递的区别在于,引用类型实参和形参之间传递内存地址,基本数据类型实参与形参之间传递值。