Java基础-值传递与引用传递

2018-06-15  本文已影响0人  Briarbear

首先,不要纠结于 Pass By Value 和 Pass By Reference 的字面上的意义,否则很容易陷入所谓的“一切传引用其实本质上是传值”这种并不能解决问题无意义论战中。更何况,要想知道Java到底是传值还是传引用,起码你要先知道传值和传引用的准确含义吧?可是如果你已经知道了这两个名字的准确含义,那么你自己就能判断Java到底是传值还是传引用。这就好像用大学的名词来解释高中的题目,对于初学者根本没有任何意义

1. 基本类型和引用类型的不同之处

int num = 10;
String str = "hello";   
image

如图所示,num是基本类型,值就直接保存在变量中。而str是引用类型,变量中保存的只是实际对象的地址。一般称这种变量为"引用",引用指向实际对象,实际对象中保存着内容。


2. 赋值运算符(=)的作用

num = 20;
str = "java";
image

对于基本类型num,赋值运算符会直接改变变量的值,原来的值被覆盖掉。

对于引用类型str,复制运算符会改变应用中所保存的地址,原来的地址别覆盖掉,但原来的对象不会被改变(重要)

如上图所示,“hello”字符串对象并没有被改变,(没有被任何引用所指向的对象是垃圾,会被垃圾回收器回收)


3. 调用方法时发生了什么

参数传递基本上就是赋值操作

 
第一个例子:基本类型
void foo(int value) {
    value = 100;
}
foo(num); // num 没有被改变
 
第二个例子:没有提供改变自身方法的引用类型
void foo(String text) {
    text = "windows";
}
foo(str); // str 也没有被改变
 
第三个例子:提供了改变自身方法的引用类型
StringBuilder sb = new StringBuilder("iphone");
void foo(StringBuilder builder) {
    builder.append("4");
}
foo(sb); // sb 被改变了,变成了"iphone4"。
 
第四个例子:提供了改变自身方法的引用类型,但是不使用,而是使用赋值运算符。
StringBuilder sb = new StringBuilder("iphone");
void foo(StringBuilder builder) {
    builder = new StringBuilder("ipad");
}
foo(sb); // sb 没有被改变,还是 "iphone"。

重点理解为什么, 第三个例子与第四个例子结果不同

image

下面是第四个例子的图解

image

4. 局部变量/方法参数

局部变量和方法参数在JVM中的存储方法是相同的,都是在栈上面开辟空间来存储的,随着进入方法开辟,退出方法回收.以32位JVM为例,boolean/byte/short/char/int/float以及引用都是分配4字节空间,long/double分配8字节空间. 对于每个方法来说,最多占用空间是一定的,这在编译时就可以计算好

我们都知道JVM内存模型中有,stack和heap的存在,但是更准确的说,是每个线程都分配一个独享的stack,所有线程共享一个heap。对于每个方法的局部变量来说,是绝对无法被其他方法,甚至其他线程的同一方法所访问到的,更遑论修改

当我们在方法中声明一个 int i = 0,或者 Object obj = null 时,仅仅涉及stack,不影响到heap,当我们 new Object() 时,会在heap中开辟一段内存并初始化Object对象。当我们将这个对象赋予obj变量时,仅仅是stack中代表obj的那4个字节变更为这个对象的地址。


5. 数组类型引用和对象

当然某些JVM实现会把"hello"字面量生成的String对象放到常量池中,而常量池中的对象可以实际分配在heap中,有些实现也许会分配在方法区,当然这对我们理解影响不大

相关链接

上一篇 下一篇

猜你喜欢

热点阅读