Java Object 引用传递和值传递

Java Object 引用传递和值传递

@author ixenosjava

 

 

 

Java没有引用传递:


除了在将参数传递给方法(或函数)的时候是"值传递",传递对象引用的副本,在任何用"="向引用对象变量赋值的时候也是传递引用的副本windows

  参数传递,传递引用的副本,这看起来是引用传递,实则是传递了副本,这已是值传递的概念了;app

  变量赋值,传递引用的副本,即对象地址值的副本iphone

 

Java参数传递中没有引用传递都是值传递 


1.在 Java 应用程序中永远不会传递对象,而只传递对象引用。所以是按引用传递对象。注意区分按引用传递对象和引用传递的区别,但重要的是要区分参数是如何传递的,这才是该节选的意图。函数

2.Java 应用程序按引用传递对象这一事实并不意味着 Java 应用程序按引用传递参数参数能够是对象引用,而 Java 应用程序是按值传递对象引用的ui

3.Java 应用程序中的变量能够为如下两种类型之一:引用类型或基本类型。spa

  看成为参数传递给一个方法时,处理这两种类型的方式是相同的。两种类型都是按值传递的;没有一种按引用传递。code

4.按值传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的一个副本对象

  基本类型传递的是值的副本,
  引用类型传递的是引用的副本。

  所以,若是函数修改了该参数,仅改变副本,而原始值保持不变。blog

 按引用传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的内存地址,是引用的副本而不是引用变量地址值的副本。

  所以,若是函数修改了该参数,调用代码中的原始值也随之改变。

  这就是引用值传递后还能经过引用调用对象内部方法的缘由

 

 

C++和Java的差别


差别只在传递形参时:

  当传递给函数的参数不是引用时,传递的都是该值的一个副本(按值传递)。

  区别在于引用。

  在 C++ 中当传递给函数的参数是引用时,您传递的就是这个引用,或者内存地址(按引用传递)。

  在 Java 应用程序中,当对象引用是传递给方法的一个参数时,您传递的是该引用的一个副本(按值传递),而不是引用自己

  Java 应用程序按值传递全部参数,这样就制做全部参数的副本,而无论它们的类型。

 

那么问题来了,Java中引用的副本实质是什么呢

  实质就是一份对象的地址,是新的引用变量(好比形参)的

  副本改变, 原数据不会改变。

  当一个引用的副本改变了引用所指向的地址中的属性的时候 外边看起来好像改变了原数据?

  其实否则!

  原数据是引用的值 即 该引用所指向的地址,也即对象的地址

  副本的值也是这个地址,故副本能够经过这个地址改变这个地址中的数据,但不可改变这个地址的值!这就是值传递的奥义

  可是若是改变了副本的值即让副本指向一个新的地址的时候这是不会改变原引用的值的

  原引用在这里是始终指向一个地址,不会随副本的改变而变

 

  具体参考:

做者:Intopass
连接:https://www.zhihu.com/question/31203609/answer/50992895
来源:知乎
著做权归做者全部,转载请联系做者得到受权。

一:搞清楚 基本类型 和 引用类型的不一样之处
int num = 10;
String str = "hello";

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

二:搞清楚赋值运算符(=)的做用
num = 20;
str = "java";
对于基本类型 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"。

重点理解为何,第三个例子和第四个例子结果不一样?

下面是第三个例子的图解:
builder.append("4")以后
下面是第四个例子的图解:

builder = new StringBuilder("ipad"); 以后
相关文章
相关标签/搜索