關于java傳參時是引用傳遞還是值傳遞,一直是一個讨論比較多的話題,
有論壇說java中隻有值傳遞,也有些地方說引用傳遞和值傳遞都存在,比較容易讓人迷惑。
關于值傳遞和引用傳遞其實需要分情況看待,今天學習和分析一下,着急可以先看最後的結論。
java中資料類型分為兩大類,基本類型和對象類型。相應的,變量也有兩種類型:基本類型和引用類型。
基本類型的變量儲存原始值,即它代表的值就是數值本身;
而引用類型的變量儲存引用值,"引用值"指向記憶體空間的位址,代表了某個對象的引用,而不是對象本身,
對象本身存放在這個引用值所表示的位址的位置。
基本類型包括:byte,short,int,long,char,float,double,boolean,returnaddress,
引用類型包括:類類型,接口類型和數組。
相應的,變量也有兩種類型:基本類型和引用類型。
基本資料類型在聲明時系統就給它配置設定空間:
1
2
<code>int</code> <code>a;</code>
<code>a=</code><code>10</code><code>;</code><code>//正确,因為聲明a時就配置設定了空間</code>
引用則不同,它聲明時隻給變量配置設定了引用空間,而不配置設定資料空間:
3
4
5
6
7
<code>date date;</code>
<code>//執行執行個體化,開辟資料空間存放date對象,然後把空間的首位址傳給today變量 </code>
<code>//date=new date();</code>
<code>//如果注釋掉上一步操作</code>
<code>//the local variable date may not have been initialized</code>
<code>//也就是說對象的資料空間沒有配置設定</code>
<code>date.getdate();</code>
看一下下面的初始化過程,注意"引用"也是占用空間的,一個空object對象的引用大小大概是4byte:
<code>date a,b; </code><code>//在記憶體開辟兩個引用空間</code>
<code>a = </code><code>new</code> <code>date();</code><code>//開辟存儲date對象的資料空間,并把該空間的首位址賦給a</code>
<code>b = a; </code><code>//将a存儲空間中的位址寫到b的存儲空間中</code>
這裡要用實際參數和形式參數的概念來幫助了解,
值傳遞:
方法調用時,實際參數把它的值傳遞給對應的形式參數,函數接收的是原始值的一個copy,此時記憶體中存在兩個相等的基本類型,即實際參數和形式參數,後面方法中的操作都是對形參這個值的修改,不影響實際參數的值。
引用傳遞:
也稱為傳位址。方法調用時,實際參數的引用(位址,而不是參數的值)被傳遞給方法中相對應的形式參數,函數接收的是原始值的記憶體位址;
在方法執行中,形參和實參内容相同,指向同一塊記憶體位址,方法執行中對引用的操作将會影響到實際對象。
看一個例子:
<code>class</code> <code>myobj{</code>
<code> </code><code>public</code> <code>int</code> <code>b=</code><code>99</code><code>;</code>
<code>}</code>
分别傳參int和對象類型:
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
<code>public</code> <code>class</code> <code>referencepkvalue2 {</code>
<code> </code>
<code> </code><code>public</code> <code>static</code> <code>void</code> <code>main(string[] args) { </code>
<code> </code><code>referencepkvalue2 t = </code><code>new</code> <code>referencepkvalue2(); </code>
<code> </code><code>int</code> <code>a=</code><code>99</code><code>; </code>
<code> </code><code>t.test1(a);</code><code>//這裡傳遞的參數a就是按值傳遞 </code>
<code> </code><code>system.out.println(a);</code>
<code> </code>
<code> </code><code>myobj obj=</code><code>new</code> <code>myobj(); </code>
<code> </code><code>t.test2(obj);</code><code>//這裡傳遞的參數obj就是引用傳遞</code>
<code> </code><code>system.out.println(obj.b);</code>
<code> </code><code>} </code>
<code> </code><code>public</code> <code>void</code> <code>test1(</code><code>int</code> <code>a){ </code>
<code> </code><code>a=a++;</code>
<code> </code><code>} </code>
<code> </code><code>public</code> <code>void</code> <code>test2(myobj obj){ </code>
<code> </code><code>obj.b=</code><code>100</code><code>;</code>
<code> </code><code>}</code>
輸出是:
99
99
100
100
可以看到,int值沒有發生變化,但是在test2方法中對obj類做的修改影響了obj這個對象。
這裡要特殊考慮string,以及integer、double等幾個基本類型包裝類,它們都是immutable類型,
因為沒有提供自身修改的函數,每次操作都是新生成一個對象,是以要特殊對待,可以認為是和基本資料類型相似,傳值操作。
看下面的例子:
<code>public</code> <code>class</code> <code>referencepkvalue1 {</code>
<code> </code><code>public</code> <code>static</code> <code>void</code> <code>main(string[] args){</code>
<code> </code><code>referencepkvalue1 pk=</code><code>new</code> <code>referencepkvalue1();</code>
<code> </code><code>//string類似基本類型,值傳遞,不會改變實際參數的值</code>
<code> </code><code>string test1=</code><code>"hello"</code><code>;</code>
<code> </code><code>pk.change(test1);</code>
<code> </code><code>system.out.println(test1);</code>
<code> </code><code>//stringbuffer和stringbuilder等是引用傳遞</code>
<code> </code><code>stringbuffer test2=</code><code>new</code> <code>stringbuffer(</code><code>"hello"</code><code>);</code>
<code> </code><code>pk.change(test2);</code>
<code> </code><code>system.out.println(test2.tostring());</code>
<code> </code><code>}</code>
<code> </code><code>public</code> <code>void</code> <code>change(string str){</code>
<code> </code><code>str=str+</code><code>"world"</code><code>;</code>
<code> </code><code>public</code> <code>void</code> <code>change(stringbuffer str){</code>
<code> </code><code>str.append(</code><code>"world"</code><code>);</code>
hello
helloworld
對string和stringbuffer的操作産生了不同的結果。
結合上面的分析,關于值傳遞和引用傳遞可以得出這樣的結論:
(1)基本資料類型傳值,對形參的修改不會影響實參;
(2)引用類型傳引用,形參和實參指向同一個記憶體位址(同一個對象),是以對參數的修改會影響到實際的對象;
(3)string, integer, double等immutable的類型特殊處理,可以了解為傳值,最後的操作不會修改實參對象。