首先,java中方法参数传递方式是按值传递。

  • 如果参数是基本类型,传递的是基本类型的字面量值的拷贝。
  • 如果参数是引用类型,传递的是该参量所引用的对象在堆中地址值的拷贝。
    这两句已经很清楚的解释了JAVA是如何传递方法参数的,即传递值的拷贝(我的理解为在调用方法者的栈帧中,拷贝一份值(不论是基本类型还是地址)到操作数栈中,然后将其推出,赋值给方法栈帧的局部变量中)

搞清楚 基本类型 和 引用类型的不同之处

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

Java方法的传值与传址 java方法值传递和引用传递_JAVA


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

搞清楚赋值运算符(=)的作用

num = 20;
str = "java";

Java方法的传值与传址 java方法值传递和引用传递_基本类型_02


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

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

如上图所示,”hello” 字符串对象没有被改变。(没有被任何引用所指向的对象是垃圾,会被垃圾回收器回收)关于gc垃圾回收需要学习

调用方法时发生了什么?参数传递基本上就是赋值操作。

//第一个例子:基本类型
void foo(int value) {
    value = 100;
}
foo(num); // num 没有被改变

//第二个例子:没有提供改变自身方法的引用类型
void foo(String text) {
    text = "windows";
}
foo(str); // str 也没有被改变 str是main()(假设他调用foo方法)方法的局部变量,他依旧指向‘java’的地址(通过常量池的引用),而在foo()栈帧中,text这个引用则在赋值运算后,指向了字符串windows的地址

//第三个例子:提供了改变自身方法的引用类型
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"。

第二个例子中, str 也没有被改变 str是main()(假设他调用foo方法)方法的局部变量,他依旧指向‘java’的地址(通过常量池的引用),而在foo()栈帧中,text这个引用则在赋值运算后,指向了字符串windows的地址。

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

下面是第三个例子的图解:

Java方法的传值与传址 java方法值传递和引用传递_传值_03


builder.append(“4”)之后

Java方法的传值与传址 java方法值传递和引用传递_引用类型_04


下面是第四个例子的图解:

Java方法的传值与传址 java方法值传递和引用传递_JAVA_05


builder = new StringBuilder(“ipad”); 之后

Java方法的传值与传址 java方法值传递和引用传递_JAVA_06