1、什么是值传递
值传递,是将内存空间中某个存储单元中存放的值,传送给另一个存储单元。(java中的存储单元并不是物理内存的地址,但具有相关性)
例如:
<code class="hljs cs has-numbering" style="display: block; padding: 0px; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-radius: 0px; word-wrap: normal; background: transparent;"><span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;">//定义了一个改变参数值的函数</span>
<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">public</span> <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">static</span> <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">void</span> <span class="hljs-title" style="box-sizing: border-box;">changeValue</span>(<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">int</span> x) {
x = x *<span class="hljs-number" style="color: rgb(0, 102, 102); box-sizing: border-box;">2</span>;
}
<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">public</span> <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">class</span> TestMain{
<span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;">//调用该函数</span>
<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">int</span> num = <span class="hljs-number" style="color: rgb(0, 102, 102); box-sizing: border-box;">5</span>;
System.<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">out</span>.println(num);
changeValue(num);
System.<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">out</span>.println(num);
}</code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right; background-color: rgb(238, 238, 238);"><li style="box-sizing: border-box; padding: 0px 5px;">1</li><li style="box-sizing: border-box; padding: 0px 5px;">2</li><li style="box-sizing: border-box; padding: 0px 5px;">3</li><li style="box-sizing: border-box; padding: 0px 5px;">4</li><li style="box-sizing: border-box; padding: 0px 5px;">5</li><li style="box-sizing: border-box; padding: 0px 5px;">6</li><li style="box-sizing: border-box; padding: 0px 5px;">7</li><li style="box-sizing: border-box; padding: 0px 5px;">8</li><li style="box-sizing: border-box; padding: 0px 5px;">9</li><li style="box-sizing: border-box; padding: 0px 5px;">10</li><li style="box-sizing: border-box; padding: 0px 5px;">11</li></ul>
结果如下:
5
5
调用函数changeValue()前后num的值都没有改变。具体过程如图:
![这里写图片描述](https://img-blog.csdn.net/20160423125941643)
- num作为参数传递给changeValue(int x)方法时,首先在内存空间中为x变量分配一个存储单元(我们说x指向这个存储单元)。
- 将内存空间中num指向的存储单元中存放的值(即”5”),传递给了changeValue(int x)中的参数变量(即”x”),也就是把”5”传给了x变量指向的存储单元中。
- changeValue(int x)方法中对x变量的一切操作,都是针对x指向的存储单元。与num指向的存储单元没有关系,当然也不会改变这个存储单元中的值。
所以,值传递,传递的是存储单元中的内容(8种基本类型:值,非基本类型:实际对象的地址)。
对于String来说JVM有他特殊的处理,了解更多看这里:String在内存中如何存放
2、什么是引用传递
java中只有值传递,没有引用传递。
所谓的引用传递,只是一个错误的概念。
例如:
<code class="hljs cs has-numbering" style="display: block; padding: 0px; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-radius: 0px; word-wrap: normal; background: transparent;">class person {
<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">public</span> <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">static</span> String name = <span class="hljs-string" style="color: rgb(0, 136, 0); box-sizing: border-box;">"Jack"</span>;
<span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;">//定义一个改变对象属性的方法 </span>
<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">public</span> <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">static</span> <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">void</span> <span class="hljs-title" style="box-sizing: border-box;">changeName</span>(Person p) {
p.name = <span class="hljs-string" style="color: rgb(0, 136, 0); box-sizing: border-box;">"Rose"</span>;
}
<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">public</span> <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">static</span> <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">void</span> <span class="hljs-title" style="box-sizing: border-box;">main</span>(String[] args) {
<span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;">//定义一个Person对象,person是这个对象的引用 </span>
Person person = <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">new</span> Person();
<span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;">//先显示这个对象的name属性 </span>
System.<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">out</span>.println(person.name);
<span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;">//调用changeName(Person p)方法 </span>
changeName(person);
<span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;">//再显示这个对象的name属性,看是否发生了变化 </span>
System.<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">out</span>.println(person.name);
}
}</code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right; background-color: rgb(238, 238, 238);"><li style="box-sizing: border-box; padding: 0px 5px;">1</li><li style="box-sizing: border-box; padding: 0px 5px;">2</li><li style="box-sizing: border-box; padding: 0px 5px;">3</li><li style="box-sizing: border-box; padding: 0px 5px;">4</li><li style="box-sizing: border-box; padding: 0px 5px;">5</li><li style="box-sizing: border-box; padding: 0px 5px;">6</li><li style="box-sizing: border-box; padding: 0px 5px;">7</li><li style="box-sizing: border-box; padding: 0px 5px;">8</li><li style="box-sizing: border-box; padding: 0px 5px;">9</li><li style="box-sizing: border-box; padding: 0px 5px;">10</li><li style="box-sizing: border-box; padding: 0px 5px;">11</li><li style="box-sizing: border-box; padding: 0px 5px;">12</li><li style="box-sizing: border-box; padding: 0px 5px;">13</li><li style="box-sizing: border-box; padding: 0px 5px;">14</li><li style="box-sizing: border-box; padding: 0px 5px;">15</li><li style="box-sizing: border-box; padding: 0px 5px;">16</li><li style="box-sizing: border-box; padding: 0px 5px;">17</li><li style="box-sizing: border-box; padding: 0px 5px;">18</li></ul>
执行后结果:
Jack
Rose
从结果看,方法用了一个对象参数,操作参数就可以改变传入对象。我们的对引用传递的错误观念这么认为:
该对象复制了一个引用副本,传给调用方法的参数,使得该方法可以对这个对象进行操作。
这种观念是初学者常犯的错误。
实际上过程如图:
![这里写图片描述](https://img-blog.csdn.net/20160423130228231)
- main方法中new了一个对象Person,存储空间中实际分配了两个对象:新创建Person类的实体对象、指向该对象的引用变量person。
其中,实体对象存放在堆内存中,引用变量存放在栈内存(Java存储特性)。
了解更多java存储看这里: - 引用变量person指向的栈内存中,存放的是堆中实体对象的逻辑地址。
- 调用changeName(Person p)方法,将person引用变量传入该方法参数p中(按照值传递,传递的是:实体对象的逻辑地址)。此时,changeName方法中对p的操作,与person没有关系。
- changeName方法中,是对p指向的存储单元中的值(即实体对象的逻辑地址)所指向的实体对象进行操作。直接改变了该实体对象。
- 由于person指向的存储单元中的值也是该实体对象的逻辑地址,这个实体对象已经在第4步中被改变了。所以有上面的结果。
3、引用传递和值传递有什么区别
引用传递是个伪概念,java中只有值传递。
【转】http://blog.csdn.net/ooppookid/article/details/51225815