Java:按值傳遞與按引用傳遞
原鏈接:傳送門
前天在做系統的時候被Java中參數傳遞問題卡了一下,回頭查閱了相關的資料,對參數傳遞問題有了新的了解和掌握,但是有個問題感覺還是很模糊,就是Java中到底是否只存在值傳遞,因為在查閱資料時,經常看到有人說Java只有值傳遞,但有人說既有值傳遞,也有引用傳遞,對於兩個觀點個人覺得應該是站的角度不同而得出兩個不同的說法,其實兩個說法其中的原理是一樣的,只要咱們懂得其中的原理,那麽至於叫什麽也就無所謂了,下面是我在網上看到的一個帖子,解釋的感覺挺全面,就轉過來,以供以後學習參考:
1:按值傳遞是什麽
指的是在方法調用時,傳遞的參數是按值的拷貝傳遞。示例如下:
public class TempTest { private void test1(int a) { //做點事情 } public static void main(String[] args) { TempTest t = new TempTest(); int a = 3; t.test1(a);//這裏傳遞的參數a就是按值傳遞 } }
按值傳遞重要特點:傳遞的是值的拷貝,也就是說傳遞後就互不相關了。
示例如下:
public class TempTest { private void test1(int a) { a = 5; System.out.println("test1方法中的a=" + a); } public static void main(String[] args) { TempTest t = new TempTest(); int a = 3; t.test1(a);//傳遞後,test1方法對變量值的改變不影響這裏的a System.out.println(”main方法中的a = ” +a); } }
運行結果是:
test1方法中的a=5 main方法中的a=3
2:按引用傳遞是什麽
指的是在方法調用時,傳遞的參數是按引用進行傳遞,其實傳遞的引用的地址,也就是變量所對應的內存空間的地址。
示例如下:
public class TempTest { private void test1(A a) { } public static void main(String[] args) { TempTest t = new TempTest(); A a = new A(); t.test1(a); //這裏傳遞的參數a就是按引用傳遞 } } class A { public int age = 0; }
3:按引用傳遞的重要特點
傳遞的是值的引用,也就是說傳遞前和傳遞後都指向同一個引用(也就是同一個內存空間)。
示例如下:
public class TempTest { private void test1(A a) { a.age = 20; System.out.println("test1方法中的age=" + a.age); } public static void main(String[] args) { TempTest t = new TempTest(); A a = new A(); a.age = 10; t.test1(a); System.out.println(”main方法中的age = ” +a.age); } } class A { public int age = 0; }
運行結果如下:
test1方法中的age=20 main方法中的age=20
4:理解按引用傳遞的過程——內存分配示意圖
要想正確理解按引用傳遞的過程,就必須學會理解內存分配的過程,內存分配示意圖可以輔助我們去理解這個過程。
用上面的例子來進行分析:
(1):運行開始,運行第8行,創建了一個A的實例,內存分配示意如下:
(2):運行第9行,是修改A實例裏面的age的值,運行後內存分配示意如下:
(3):運行第10行,是把main方法中的變量a所引用的內存空間地址,按引用傳遞給test1方法中的a變量。請註意:這兩個a變量是完全不同的,不要被名稱相同所蒙蔽。
內存分配示意如下:
由於是按引用傳遞,也就是傳遞的是內存空間的地址,所以傳遞完成後形成的新的內存示意圖如下:
也就是說:是兩個變量都指向同一個空間。
(4):運行第3行,為test1方法中的變量a指向的A實例的age進行賦值,完成後形成的新的內存示意圖如下:
此時A實例的age值的變化是由test1方法引起的
(5):運行第4行,根據此時的內存示意圖,輸出test1方法中的age=20
(6):運行第11行,根據此時的內存示意圖,輸出main方法中的age=20
5:對上述例子的改變
理解了上面的例子,可能有人會問,那麽能不能讓按照引用傳遞的值,相互不影響呢?就是test1方法裏面的修改不影響到main方法裏面呢?
方法是在test1方法裏面新new一個實例就可以了。改變成下面的例子,其中第3行為新加的:
public class TempTest { private void test1(A a) { a = new A();//新加的一行 a.age = 20; System.out.println("test1方法中的age=" + a.age); } public static void main(String[] args) { TempTest t = new TempTest(); A a = new A(); a.age = 10; t.test1(a); System.out.println(”main方法中的age = ” +a.age); } } class A { public int age = 0; }
運行結果為:
test1方法中的age=20 main方法中的age=10
為什麽這次的運行結果和前面的例子不一樣呢,還是使用內存示意圖來理解一下
6:再次理解按引用傳遞
(1):運行開始,運行第9行,創建了一個A的實例,內存分配示意如下:
(2):運行第10行,是修改A實例裏面的age的值,運行後內存分配示意如下:
(3):運行第11行,是把main方法中的變量a所引用的內存空間地址,按引用傳遞給test1方法中的a變量。請註意:這兩個a變量是完全不同的,不要被名稱相同所蒙蔽。
內存分配示意如下:
由於是按引用傳遞,也就是傳遞的是內存空間的地址,所以傳遞完成後形成的新的內存示意圖如下:
也就是說:是兩個變量都指向同一個空間。
(4):運行第3行,為test1方法中的變量a重新生成了新的A實例的,完成後形成的新的內存示意圖如下:
(5):運行第4行,為test1方法中的變量a指向的新的A實例的age進行賦值,完成後形成的新的內存示意圖如下:
註意:這個時候test1方法中的變量a的age被改變,而main方法中的是沒有改變的。
(6):運行第5行,根據此時的內存示意圖,輸出test1方法中的age=20
(7):運行第12行,根據此時的內存示意圖,輸出main方法中的age=10
7:說明
“在Java裏面參數傳遞都是按值傳遞”這句話的意思是:按值傳遞是傳遞的值的拷貝,按引用傳遞其實傳遞的是引用的地址值,所以統稱按值傳遞。
Java:按值傳遞與按引用傳遞