Java:按值傳遞還是按引用傳遞詳細解說


1:按值傳遞是什么

指的是在方法調用時,傳遞的參數是按值的拷貝傳遞。示例如下:

 1 public class TempTest {  
 2 private void test1(int a){  
 3 //做點事情  
 4 }  
 5 public static void main(String[] args) {  
 6 TempTest t = new TempTest();  
 7 int a = 3;  
 8 t.test1(a);//這里傳遞的參數a就是按值傳遞  
 9 }  
10 }  

按值傳遞重要特點:傳遞的是值的拷貝,也就是說傳遞后就互不相關了。

示例如下:

 1 public class TempTest {  
 2 private void test1(int a){  
 3 a = 5;  
 4 System.out.println("test1方法中的a="+a);  
 5 }  
 6 public static void main(String[] args) {  
 7 TempTest t = new TempTest();  
 8 int a = 3;  
 9 t.test1(a);//傳遞后,test1方法對變量值的改變不影響這里的a  
10 System.out.println(”main方法中的a=”+a);  
11 }  
12 }  

運行結果是:

test1方法中的a=5  
main方法中的a=3 

 

 

 

2:按引用傳遞是什么

指的是在方法調用時,傳遞的參數是按引用進行傳遞,其實傳遞的引用的地址,也就是變量所對應的內存空間的地址。

示例如下:

 1 public class TempTest {  
 2 private void test1(A a){  
 3 }  
 4 public static void main(String[] args) {  
 5 TempTest t = new TempTest();  
 6 A a = new A();  
 7 t.test1(a); //這里傳遞的參數a就是按引用傳遞  
 8 }  
 9 }  
10 class A{  
11 public int age = 0;  
12 }  

 

 

3:按引用傳遞的重要特點

傳遞的是值的引用,也就是說傳遞前和傳遞后都指向同一個引用(也就是同一個內存空間)。

示例如下:

 1 public class TempTest {  
 2 private void test1(A a){  
 3 a.age = 20;  
 4 System.out.println("test1方法中的age="+a.age);  
 5 }  
 6 public static void main(String[] args) {  
 7 TempTest t = new TempTest();  
 8 A a = new A();  
 9 a.age = 10;  
10 t.test1(a);  
11 System.out.println(”main方法中的age=”+a.age);  
12 }  
13 }  
14 class A{  
15 public int age = 0;  
16 }  

運行結果如下:

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行為新加的:

 1 public class TempTest {  
 2  private void test1(A a){  
 3  a = new A();//新加的一行  
 4  a.age = 20;  
 5  System.out.println("test1方法中的age="+a.age);  
 6  }  
 7  public static void main(String[] args) {  
 8  TempTest t = new TempTest();  
 9  A a = new A();  
10  a.age = 10;  
11  t.test1(a);  
12  System.out.println(”main方法中的age=”+a.age);  
13  }  
14 }  
15 class A{  
16  public int age = 0;  
17 }  

運行結果為:

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:說明

(1):“在Java里面參數傳遞都是按值傳遞”這句話的意思是:按值傳遞是傳遞的值的拷貝,按引用傳遞其實傳遞的是引用的地址值,所以統稱按值傳遞。

(2):在Java里面只有基本類型和按照下面這種定義方式的String是按值傳遞,其它的都是按引用傳遞。就是直接使用雙引號定義字符串方式:String str = “Java私塾”;


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM