Java是值传递还是引用传递?

149 阅读3分钟
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中只有值传递,基本数据类型存在栈中,而引用数据类型,如各种对象等存在的是堆中,变量名存放的是对象的地址,所以当我们把值传递给形参的时候,是复制了一份原始数据的地址,归根结底还是值传递。