——reference Java is Pass by Value and Not Pass by Reference
其实这个问题是一个非常初级的问题,相关的概念初学者早已掌握,但是时间长了还是容易混淆,特此总结一下
一、值传递和引用传递
首先这里我们先看下两者的异同:
- 值传递:方法调用时,实际参数把它的值传递给对应的形式参数,方法执行中形式参数值的改变不影响实际参 数的值。
- 引用传递:也称为传地址。方法调用时,实际参数的引用(地址,而不是参数的值)被传递给方法中相对应的形式参数,在方法执行中,对形式参数的操作实际上就是对实际参数的操作,方法执行中形式参数值的改变将会影响实际参数的值。
二、Java总是值传递而非引用传递
举例:
Balloon.java
package com.journaldev.test; public class Balloon { private String color; public Balloon(){} public Balloon(String c){
this.color=c;
} public String getColor() {
return color;
} public void setColor(String color) {
this.color = color;
}
}
之后我们做一个简单测试,交换两个对象
Test.java
package com.journaldev.test; public class Test { public static void main(String[] args) { Balloon red = new Balloon("Red"); //memory reference 50
Balloon blue = new Balloon("Blue"); //memory reference 100 swap(red, blue);
System.out.println("red color="+red.getColor());
System.out.println("blue color="+blue.getColor()); foo(blue);
System.out.println("blue color="+blue.getColor()); } private static void foo(Balloon balloon) { //baloon=100
balloon.setColor("Red"); //baloon=100
balloon = new Balloon("Green"); //baloon=200
balloon.setColor("Blue"); //baloon = 200
} //Generic swap method
public static void swap(Object o1, Object o2){
Object temp = o1;
o1=o2;
o2=temp;
}
}
输出如下:
red color=Red
blue color=Blue
blue color=Red
从上面的代码可以得出:
- 两个对象在被调函数中交换并不能影响主调函数中的两个对象
- 但是在被调函数中对对象进行修改(对象的某一域),那么在主调函数中是可以反映出来的
第一点我们就可以看出Java确实是值传递的,然而第二点又让我们出现疑惑,为何对象的修改会影响到主调函数
这涉及到Java中的数据存储方式,堆栈存储在(Java 堆内存与栈内存异同(Java Heap Memory vs Stack Memory Difference)
Java中除了基本类型(【Java心得总结一】Java基本类型和包装类型解析)外,其他类型声明都是在栈中保存一个指向堆中的引用,而对象的数据都是在堆中保存的,如图:
所以在函数调用时,如果传递对象,Java确实是值传递,开辟一个新的存储空间存放形参,并将实参的值传递过来,但是两者指向同一处堆的存储空间,所以造成了上述交换无用,但是改变对象中的数据有用的结果