關於Java中形參與實參的理解
今天閱讀了一個寫的非常棒的博文,通過此博文再次複習了Java中引數傳遞的知識(即值傳遞與引用傳遞的區別)。參考網站http://www.cnblogs.com/binyue/p/3862276.html。下面我將通過次博文記錄自己的理解,還望能夠幫助再學Java或者複習Java的朋友們。
理解Java中的引用傳遞和值傳遞
1.基本型別和引用型別在記憶體中的儲存
Java中資料型別分為兩大類,基本型別和物件型別。相應的,變數也有兩種型別:基本型別和引用型別。
基本型別的變數儲存原始值,即它代表的值就是數值本身;
而引用型別的變數儲存引用值,"引用值"指向記憶體空間的地址,代表了某個物件的引用,而不是物件本身,
物件本身存放在這個引用值所表示的地址的位置。
基本型別包括:byte,short,int,long,char,float,double,Boolean,returnAddress,
引用型別包括:類型別,介面型別和陣列。
相應的,變數也有兩種型別:基本型別和引用型別。
2.變數的基本型別和引用型別的區別
基本資料型別在宣告時系統就給它分配空間:
1 2 |
|
引用則不同,它宣告時只給變數分配了引用空間,而不分配資料空間:
1 2 3 4 5 6 7 |
|
看一下下面的初始化過程,注意"引用"也是佔用空間的,一個空Object物件的引用大小大概是4byte:
1 2 3 |
//開闢儲存Date物件的資料空間,並把該空間的首地址賦給a
|
3.引用傳遞和值傳遞
這裡要用實際引數和形式引數的概念來幫助理解,
值傳遞:
方法呼叫時,實際引數把它的值傳遞給對應的形式引數,函式接收的是原始值的一個copy,此時記憶體中存在兩個相等的基本型別,即實際引數和形式引數,後面方法中的操作都是對形參這個值的修改,不影響實際引數的值。
引用傳遞:
也稱為傳地址。方法呼叫時,實際引數的引用(地址,而不是引數的值)被傳遞給方法中相對應的形式引數,函式接收的是原始值的記憶體地址;
在方法執行中,形參和實參內容相同,指向同一塊記憶體地址,方法執行中對引用的操作將會影響到實際物件。
看一個例子:
1 2 3 |
|
分別傳參int和物件型別:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 |
|
輸出是:
99
99
100
100
可以看到,int值沒有發生變化,但是在test2方法中對obj類做的修改影響了obj這個物件。
這裡要特殊考慮String,以及Integer、Double等幾個基本型別包裝類,它們都是immutable型別,
因為沒有提供自身修改的函式,每次操作都是新生成一個物件,所以要特殊對待,可以認為是和基本資料型別相似,傳值操作。
看下面的例子:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 |
|
輸出是:
Hello
Helloworld
對String和StringBuffer的操作產生了不同的結果。
4.結論
結合上面的分析,關於值傳遞和引用傳遞可以得出這樣的結論:
(1)基本資料型別傳值,對形參的修改不會影響實參;
(2)引用型別傳引用,形參和實參指向同一個記憶體地址(同一個物件),所以對引數的修改會影響到實際的物件;
(3)String, Integer, Double等immutable的型別特殊處理,可以理解為傳值,最後的操作不會修改實參物件。