java 是值传递(pass by value),还是引用传递(pass by reference)?
答案是:值传递(pass by value)
先说一下值传递和引用传递的概念和区别吧
值传递:是指在调用函数时,将原始参对象复制一份作为实参传给形参,当在函数中对参数进行修改时,不会影响到实际参数。
引用传递:是指在调用函数时,将原始对象直接作为实参传给形参,一旦在函数中对其进行修改,将会影响实际参数。
1、举例说明
基本类型作为参数传递 public class ValuePass {
public static void main(String[] args) {
//值传递举例
int num = 10;
System.out.println("改之前的值:" + num);
modify(num);
System.out.println("改之后的值:" + num);
}
private static void modify(int num2) {
num2 = 11;
}
}
输出结果为:
改之前的值:10
改之后的值:10
通过这个例子,说明基本数据类型作为参数传递时,传递的是值的拷贝,无论怎么改变这个拷贝,原值是不会改变的。
对象作为参数传递。
下面我们通过一个例子来说明 public class ReferenceBasicPass {
static class TreeNode {
int val;
TreeNode left;
TreeNode right;
public TreeNode(int x) { val = x; }
public void setVal(int val) {
this.val = val;
}
public int getVal() {
return val;
}
}
public static void main(String[] args) {
//普通对象
TreeNode node = new TreeNode(10);
System.out.println("实参 node 指向的内存地址为:" + node.hashCode());
System.out.println("改之前的值:" + node.getVal());
modify(node);
System.out.println("改之后的值:" + node.getVal());
}
private static void modify(TreeNode node) {
System.out.println("形参 node 指向的内存地址为:" + node.hashCode());
//引用了同一块地址,操作了同一块堆内存
node.setVal(11);
}
输出结果为:
实参 node 指向的内存地址为:366712642
改之前的值:10
形参 node 指向的内存地址为:366712642
改之后的值:11
我们发现,实际参数a的内容随着函数里的修改也发生了改变,这是不是说明对于引用类型,是引用传递呢,不,希望你们不要产生这样的误解。对于引用类型,依旧是值传递,先看看下面一段代码。
public static void main(String[] args) {
A a = new A();
a.setName("Martina");
modify(a);
System.out.println("print in main : a.name = " + a.getName());
}
public static void modify(A b) {
b = new A();
b.setName("Leon");
System.out.println("print in modify : b.name = " + b.getName());
}
看看打印结果:
print in modify : b.name = Leon
print in main : a.name = Martina
这个时候我们的原始对象并没有发生改变,而这一段与上一段的区别在于:
b = new A();
如果是引用传递,那么我们的原始参数a应该还是会发生影响的,所以还是先让我们看一下这两个例子到底发生了什么:
当我们在传递参数时,将原始对象的地址传给了形参,在函数中,对形参的内容进行了改变,也就是,改变了原始对象的内容。
而在 b=new A(); 的时候,改变了形参的地址,对原始对象的内容没有进行影响。故而我们可以判断出,这并不是引用传递,如果是,在 b=newA(); 的时候,a的地址也应该发生改变,事实并没有,故而证明java就是值传递。
还有一个比较生动的例子来描述值传递和引用传递。
总结
java中只有值传递,基本数据类型存在栈中,而引用数据类型,如各种对象等存在的是堆中,变量名存放的是对象的地址,所以当我们把值传递给形参的时候,是复制了一份原始数据的地址,归根结底还是值传递。