Java Object 引用传递和值传递
@author ixenos
Java没有引用传递:
除了在将参数传递给方法(或函数)的时候是"值传递",传递对象引用的副本,在任何用"="向引用对象变量赋值的时候也是传递引用的副本
参数传递,传递引用的副本,这看起来是引用传递,实则是传递了副本,这已经是值传递的概念了;
变量赋值,传递引用的副本,即对象地址值的副本
Java参数传递中没有引用传递都是值传递
1.在 Java 应用程序中永远不会传递对象,而只传递对象引用。因此是按引用传递对象。注意区分按引用传递对象和引用传递的区别,但重要的是要区分参数是如何传递的,这才是该节选的意图。
2.Java 应用程序按引用传递对象这一事实并不意味着 Java 应用程序按引用传递参数。参数可以是对象引用,而 Java 应用程序是按值传递对象引用的。
3.Java 应用程序中的变量可以为以下两种类型之一:引用类型或基本类型。
当作为参数传递给一个方法时,处理这两种类型的方式是相同的。两种类型都是按值传递的;没有一种按引用传递。
4.按值传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的一个副本。
基本类型传递的是值的副本,
引用类型传递的是引用的副本。
因此,如果函数修改了该参数,仅改变副本,而原始值保持不变。
按引用传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的内存地址,是引用的副本而不是引用变量地址值的副本。
因此,如果函数修改了该参数,调用代码中的原始值也随之改变。
这就是引用值传递后还能通过引用调用对象内部方法的原因
C++和Java的差异
差异只在传递形参时:
当传递给函数的参数不是引用时,传递的都是该值的一个副本(按值传递)。
区别在于引用。
在 C++ 中当传递给函数的参数是引用时,您传递的就是这个引用,或者内存地址(按引用传递)。
在 Java 应用程序中,当对象引用是传递给方法的一个参数时,您传递的是该引用的一个副本(按值传递),而不是引用本身。
Java 应用程序按值传递所有参数,这样就制作所有参数的副本,而不管它们的类型。
那么问题来了,Java中引用的副本实质是什么呢?
实质就是一份对象的地址,是新的引用变量(比如形参)的值!
副本改变, 原数据不会改变。
当一个引用的副本改变了引用所指向的地址中的属性的时候 外边看起来好像改变了原数据?
其实不然!
原数据是引用的值 即 该引用所指向的地址,也即对象的地址
副本的值也是这个地址,故副本可以通过这个地址改变这个地址中的数据,但不可改变这个地址的值!这就是值传递的奥义
但是如果改变了副本的值即让副本指向一个新的地址的时候这是不会改变原引用的值的
原引用在这里是始终指向一个地址,不会随副本的改变而变
具体参考:
作者:Intopass
链接:https://www.zhihu.com/question/31203609/answer/50992895
来源:知乎
著作权归作者所有,转载请联系作者获得授权。一:搞清楚 基本类型 和 引用类型的不同之处int num = 10;
String str = "hello";<img src="https://pic2.zhimg.com/166032bc90958c21604110441ad03f45_b.jpg" data-rawwidth="728" data-rawheight="458" class="origin_image zh-lightbox-thumb" width="728" data-original="https://pic2.zhimg.com/166032bc90958c21604110441ad03f45_r.jpg">如图所示,num是基本类型,值就直接保存在变量中。而str是引用类型,变量中保存的只是实际对象的地址。一般称这种变量为"引用",引用指向实际对象,实际对象中保存着内容。
如图所示,num是基本类型,值就直接保存在变量中。而str是引用类型,变量中保存的只是实际对象的地址。一般称这种变量为"引用",引用指向实际对象,实际对象中保存着内容。二:搞清楚赋值运算符(=)的作用
num = 20;
str = "java";<img src="https://pic3.zhimg.com/287c0efbb179638cf4cf27cbfdf3e746_b.jpg" data-rawwidth="714" data-rawheight="572" class="origin_image zh-lightbox-thumb" width="714" data-original="https://pic3.zhimg.com/287c0efbb179638cf4cf27cbfdf3e746_r.jpg">对于基本类型 num ,赋值运算符会直接改变变量的值,原来的值被覆盖掉。
对于基本类型 num ,赋值运算符会直接改变变量的值,原来的值被覆盖掉。
对于引用类型 str,赋值运算符会改变引用中所保存的地址,原来的地址被覆盖掉。但是原来的对象不会被改变(重要)。
如上图所示,"hello" 字符串对象没有被改变。(没有被任何引用所指向的对象是垃圾,会被垃圾回收器回收)三:调用方法时发生了什么?参数传递基本上就是赋值操作。
第一个例子:基本类型
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"。重点理解为什么,第三个例子和第四个例子结果不同?
下面是第三个例子的图解:
<img src="https://pic4.zhimg.com/d8b82e07ea21375ca6b300f9162aa95f_b.jpg" data-rawwidth="772" data-rawheight="398" class="origin_image zh-lightbox-thumb" width="772" data-original="https://pic4.zhimg.com/d8b82e07ea21375ca6b300f9162aa95f_r.jpg">builder.append("4")之后builder.append("4")之后
<img src="https://pic4.zhimg.com/ff2ede9c6c55568d42425561f25a0fd7_b.jpg" data-rawwidth="696" data-rawheight="424" class="origin_image zh-lightbox-thumb" width="696" data-original="https://pic4.zhimg.com/ff2ede9c6c55568d42425561f25a0fd7_r.jpg">下面是第四个例子的图解:下面是第四个例子的图解:
<img src="https://pic4.zhimg.com/d8b82e07ea21375ca6b300f9162aa95f_b.jpg" data-rawwidth="772" data-rawheight="398" class="origin_image zh-lightbox-thumb" width="772" data-original="https://pic4.zhimg.com/d8b82e07ea21375ca6b300f9162aa95f_r.jpg">
builder = new StringBuilder("ipad"); 之后
<img src="https://pic2.zhimg.com/46fa5f10cc135a3ca087dae35a5211bd_b.jpg" data-rawwidth="710" data-rawheight="438" class="origin_image zh-lightbox-thumb" width="710" data-original="https://pic2.zhimg.com/46fa5f10cc135a3ca087dae35a5211bd_r.jpg">