先给出肯定答案:Java是值传递。
在java中,无论是传递基本类型参数还是引用类型参数,都是值传递,只是在传递基本类型时,参数是基本类型数值的副本,在传递引用类型时,参数是引用的值的副本。听起来很抽象?下面结合代码来说明。
首先,请考虑下面代码的运行结果:
public class Reference {
/**
* ecustyuanzhen
* @param args
*/
public static void main(String[] args) {
// TODO Auto-generated method stub
int i = 0;
People p = new People();
System.out.println(p.id+","+i);
change(p,i);
System.out.println(p.id+","+i);
}
static void change(People people,int a){
people.id=2;
a=2;
}
}
class People{
public int id = 0;
}
out:0,0
2,0
对于第一行输出肯定毫无异议。
分析第二行,people的属性竟然变了而且被保存了。方法内的局部变量,毫无变化。
当程序执行
change(p,i);
后,该方法实际接收到的参数people、a的值如下所示:
基本类型直接是值的副本,所以直接在栈上产生一个副本它的值为0。
引用类型people,是原p引用的副本,所以相当于语句 people=p;即people的值和p的值一样,也就是指向的堆内存对象一致。
现在执行方法体里面语句:
people.id=2;
a=2;
如图:
此时,a的值和堆内存中id值均发生变化。
接着方法执行结束,方法里的局部变量被回收(即people和a)。如图:
程序打印
System.out.println(p.id+","+i);
输出:2,0
乘热打铁,再考虑下面代码:
public class Reference {
/**
* ecustyuanzhen
* @param args
*/
public static void main(String[] args) {
// TODO Auto-generated method stub
People p = new People();
System.out.println(p.id);
change(p);
System.out.println(p.id);
}
static void change(People people){
People p = new People();
people = p;
people.id=2;
}
}
class People{
public int id = 0;
}
out:0
0
直接上图:
在调用方法时和上面类似,首先引用复制,但是代码体实现不同:
People p = new People();
people = p;
方法内重新初始化一个新对象,并赋值给传进来的参数people,然后将该对象进行修改。
在方法结束后,局部变量被清理。
所以,原对象的属性毫无变化。
结论:Java中只有值传递。只是在传递基本类型和应用类型时处理上有点不同,基本类型传递的是真正的值的副本,引用类型传递的是引用对应对象的变量的值的副本。