要想知道是引用传递还是值传递,你首先就要明白,什么是值传递,什么又是引用传递。
首先是:"引用传递"(pass by reference)是编程中的一个概念,特别是在参数传递中。在引用传递中,函数的参数是通过引用来传递的,而不是通过值来传递。这意味着,在函数内部对参数的修改会影响到原始数据。
其次是:值传递(pass by value)是计算机编程中的一个概念,特别是在参数传递中。在值传递中,函数接收的是参数值的副本,对参数的任何修改都不会影响原始数据。这与引用传递(pass by reference)相对,在引用传递中,函数接收的是参数的引用,对参数的修改会直接影响原始数据。
那么值传递和应用传递的区别是什么呢?
在我看来,值传递和引用传递最大的区别就是:所谓的值传递,就是在传递值的时候,他会进行复制,但是他没有办法去改变原本的值。而引用传递,他所得到是数据的地址值,他不会进行拷贝数据本身的值。地址值是在站里面,进而指向堆里面数据。
两者的最主要区别就是是直接传递的,还是传递的是一个副本
public static void main(String[] args) { String str = "hello"; change(str); System.out.println(); } private static void change(String str) { str = "world"; }
上面这个代码,如果你觉得基本数据类型是值传递,引用数据类型是引用传递的话,那么应该输出world对吧,但是你去执行的话,输出的数是hello。
分析这个问题,我们需要知道变量在jvm中是怎么存储的。首先看基本类型,这个很简单,变量在栈中直接存的是值,传到方法的是这个变量的拷贝,因此对拷贝的变量修改不会影响原变量的值。接着看引用类型,变量在栈中存储的是引用地址,这个地址指向堆中具体的值当调用方法传入变量时,也是拷贝变量,但是这里的拷贝只是栈中的引用地址,并不会拷贝堆中的数据,因此会变原本数据和copy数据的地址值是一样的,大家都指向一个数据。虽然变量是拷贝,但是指向的地址是同一个,因此对变量中的数据修改时,还是会影响到原来真实的变量,但是,如果我们修改的是变量在栈中的地址,则不会影响原变量。
public static void main(String[] args) { Student student = new Student(); student.setAge(18); changeAge(); System.out.println(student.getAge()); } public static void changeAge(){ Student student = new Student(); student.setAge(19); }
当你修改变量的地址值,去创建一个新的student的时候,原本的值就不会发生改变
下面是不创建的时候
public static void main(String[] args) { Student student = new Student(); student.setAge(18); changeAge(student); System.out.println(student.getAge()); } public static void changeAge(Student student){ // Student student = new Student(); student.setAge(19); } }
输出的值就为19。
那String明明没有进行新的创建,为什么值也变了呢?
当你使用字面量(如 String str = "hello";
)创建一个String对象时,Java会首先查看字符串常量池(String Constant Pool)中是否已经存在一个与该字面量相同的字符串。如果存在,则返回对该已存在字符串的引用;如果不存在,则在字符串常量池中创建一个新的String对象,并返回对它的引用。如果你使用 new
关键字来创建String对象(如 String str = new String("hello");
),无论字符串常量池中是否已存在该字符串,都会在堆内存中创建一个新的String对象。同时,如果字符串常量池中已存在该字符串,新创建的对象会包含与池中字符串相同的字符数组,但它们是两个不同的对象。