Java值传递:
是针对Java基本类型(int等)数据的传递,在方法执行完以后,形参值的改变不会影响到实参的数据,这个没什么好说的。
Java引用传递:
首先需要明确一点,引用的含义,引用是个通用名词,在C++中表示变量的别名,而在Java中,引用类似于C++的指针,但又不太相同,因为Java的引用并不需要你自己去控制对象的使用周期,由于Jvm自带垃圾回收,所以该引用是是对象在jvm中的内存地址,Java为了避免出现内存泄露问题,出于安全性以及GC回收机制,也不允许对该变量进行自增操作,而是让Java程序直接使用该内存地址的内容,C++中的指针是可以的。
所以有很多人说,Java并没有引用传递,也是对的,只不过他们把此处的引用当做了C++的引用传递而已。
我们姑且在Java中传递地址的方式称为引用传递,那么Java到底是不是传递的地址呢,看示例。

public class User {

    private String name;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

}
public class Test {

    public void set(User user){
        user.setName("hello world");
    }

    public static void main(String[] args) {

        Test test = new Test();
        User user = new User();
        test.set(user);
        System.out.println(user.getName());
    }
}

上面代码的输出结果是“hello world”,这不必多说,那如果将set方法改为如下,结果会是多少呢?

public void set(User user){
        user.setName("hello world");
        user = new User();
        user.setName("change");
    }

答案依然是“hello world”,下面就让我们来分析一下如上代码。
首先

User user = new User();

是在堆中创建了一个对象,并在栈中创建了一个引用,此引用指向该对象,如下图:

java值传递引用传递 java 引用传递和值传递_java

test.set(user);

是将引用user作为参数传递到set方法,注意:这里传递的并不是引用本身,而是一个引用的拷贝。也就是说这时有两个引用(引用和引用的拷贝)同时指向堆中的对象,如下图:

java值传递引用传递 java 引用传递和值传递_User_02

user = new User();

在set()方法中,又创建了一个User对象,并将“user引用的拷贝”指向这个在堆中新创建的对象,如下图:

java值传递引用传递 java 引用传递和值传递_java_03

user.setName("change");

在set()方法中,“user引用的拷贝”操作的是堆中新创建的User对象。

java值传递引用传递 java 引用传递和值传递_Java_04

System.out.println(user.getName());

set()方法执行完毕,目光再回到mian()方法,因为之前,”user引用的拷贝”已经将堆中的User对象的name属性设置为了”hello world”,所以当main()方法中的user调用getName()时,打印的结果就是”hello world”。如下图:

java值传递引用传递 java 引用传递和值传递_引用_05


Java虚拟机体系结构