Java基础知识详解:值传递

@author:Tobin
水平有限,如有错误,望请斧正。
参考《Java核心技术卷-基础知识第10版》
结合博文:https://juejin.im/post/5bce68226fb9a05ce46a0476,对其进行总结。

首先问自己什么是引用传递,什么是值传递?
值传递:将副本传递给方法,调用方法改变副本的值,但是并不改变原值
引用传递:传递的是对象(或者变量)的引用,对其修改,会改变原值。

从内存分配的角度讲解Java中只存在值传递,不存在引用传递
Java中方法参数的使用情况。
1.一个方法不能够修改基本数据类型的参数。
原因在于基本数据类型参数比如main方法中的参数,是存储在自身的栈帧中的,当调用方法时,会相应的建立新的栈帧,拷贝传入的参数,调用方法所做的修改也只是针对拷贝后的参数进行的,当离开这个函数,方法栈帧被销毁,被修改的参数也会被销毁。回到main栈帧中,原来的变量并没有被修改。可以看下面的这张图理解。

2.一个方法可以改变一个对象参数的状态。
下面这张图,per就是一个对象,它的值是一块地址区域(当然和c++中的纯地址有区别)。这个地址区域指向堆中的一块区域。new创建的对象都是存储在堆中的。当我们创建一个对象时,其实是两步。

Employee per;
per=new Employee();

第一步就是虚拟机栈中分配的区域,第二步就是在堆中分配区域。
如何理解没有引用传递呢?我们知道如果传入的参数是对象,方法对对象的修改会改变原来的对象值。原因在于,我们传入的参数per,此时依然是拷贝了新的变量放在方法栈帧中,它和原来的per值是相同的,但是对它进行修改,其实是对指向的堆中元素修改。堆是共享的,并不会随着方法栈帧的销毁而销毁。所以per的值就会被改变。
所以在java中无论是对象参数还是基本类型参数,它们的传递就是拷贝新的值,是值传递。
下面这段代码,进一步验证了对象参数是值传递,a,b对象并没有交换它们所指的对象,因为传入的参数仅仅是它们值得拷贝,换句话说它们的值并没有被改变,但是如果改变指向的堆区的状态是可以的。

public static void swap(Employee x , Employee y) // doesn't work
    Employee temp = x;
    x = y;
    y = temp;
} 
Employee a = new Employee("Alice", . . .);
Employee b = new Employee("Bob", . . .);
swap(a, b);
// does a now refer to Bob, b to Alice? 

这点和C++不同,C++是有引用传递的。
综上,即使改变了指向的堆的状态,原来的对象对应的地址值也并没有被修改。根据开始时所说的引用传递的定义,我们可以得到结论,Java中只有值传递,并没有引用传递。这就造成了直接swap(a,b)是无法交换两个变量的值的。知道这个结论后,我们应该明白有些操作是做不了的,而不是死记概念。

下面这幅图展现了Java程序的加载与存储,原博讲解的不错,注意一下可以看到static方法是存放在一个特定的区域的,它不属于对象,属于类。

原文地址:https://www.cnblogs.com/zuotongbin/p/11091026.html