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

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

首先我们先了解下值传递和引用传递的基本概念:

值传递(pass by value)是指在调用函数时将实际参数复制一份传递到函数中,
这样在函数中如果对参数进行修改,将不会影响到实际参数。


引用传递(pass by reference)是指在调用函数时将实际参数的地址直接传递到
函数中,那么在函数中对参数所进行的修改,将影响到实际参数。

两种方式根本区别就在于是否创建副本,所导致结果的区别为是否会直接操作原始对象。

首先基本类型 :

package com.juc.demo.test;

public class aTest {
          
   
    static void  change(int value) {
          
   
        value = 100;
    }
    public static void main(String[] args) {
          
   
        int num=2;
        change(num);
        System.out.println("after change num = " +num);
    }
}

输出结果 为2,没有改变原始值,不管我们在change()方法里面对value如何操作,都不会改变num的值,很明显是值传递。

after change num = 2

对于引用类型来说,我们通过两个例子来说明:

第一个例子:

package com.juc.demo.test;

public class aTest {
          
   

    static void  change(User user) {
          
   
        user.setName("李四同学");
    }
    public static void main(String[] args) {
          
   
        User user= new User();
        user.setName("张三同学");
        change(user);
        System.out.println("after change name = " +user.getName());
    }

}
class User{
          
   
    private String name;

    public String getName() {
          
   
        return name;
    }

    public void setName(String name) {
          
   
        this.name = name;
    }
}
after change name = 李四同学

第二个例子:

package com.juc.demo.test;

public class aTest {
          
   

    static void  change(User user) {
          
   
        user=new User();
        user.setName("李四同学");
    }
    public static void main(String[] args) {
          
   
        User user= new User();
        user.setName("张三同学");
        change(user);
        System.out.println("after change name = " +user.getName());
    }

}
class User{
          
   
    private String name;

    public String getName() {
          
   
        return name;
    }

    public void setName(String name) {
          
   
        this.name = name;
    }
}
after change name = 张三同学

我们可以看到,第一个例子结果改变了原始对象的name,第二个没有改变,为什么呢? 假设原user对象的地址是0x123456,其实,不管是哪个例子,java都有复制一个新的副本,保存的是user对象的地址0x123456,我们把这个复制的地址传进change()方法里。

    第一个例子,虽然创建了引用地址的副本,由于该地址副本指向的仍是堆中的user对象,所以对该对象操作会被改变; 第二个例子中,user=new User(); 开辟了新的堆空间,假设地址是0x567899,使得副本指向该地址,然后再对该地址的user进行操作,也不会影响到原user对象。

如果java是引用传递,即直接将该地址传递过去,而没有拷贝新的副本,那第二个例子中参数会指向新的地址,对新地址的user操作,结果一定会改变。

所以,不管是基本类型的数据,还是引用类型,java都是拷贝了新的副本传值,只不过基本数据类型拷贝的是数据本身,而引用类型拷贝的是引用地址,并非直接把该地址传进去。两种方式根本区别就在于是否创建副本,而不是说传递的类型是值还是引用。 总而言之java只有值传递,不存在引用传递。

经验分享 程序员 微信小程序 职场和发展