程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> JAVA編程 >> JAVA編程入門知識 >> Java中的值傳遞和引用傳遞

Java中的值傳遞和引用傳遞

編輯:JAVA編程入門知識

  前幾天面試被問到這個問題,並且在項目中也曾經遇到過類似的問題,由於沒有做總結,導致回答的一塌糊塗。在網上查資料,終於一下子情況了許多。

  當一個對象被當作參數傳遞到一個方法後,此方法可改變這個對象的屬性,並可返回變化後的結果,那麼這裡到底是值傳遞還是引用傳遞? 

  答:是值傳遞。Java 編程語言只有值傳遞參數。當一個對象實例作為一個參數被傳遞到方法中時,參數的值就是該對象的引用一個副本。指向同一個對象,對象的內容可以在被調用的方法中改變,但對象的引用(不是引用的副本)是永遠不會改變的。

  Java參數,不管是原始類型還是引用類型,傳遞的都是副本(有另外一種說法是傳值,但是說傳副本更好理解吧,傳值通常是相對傳址而言)。

      如果參數類型是原始類型,那麼傳過來的就是這個參數的一個副本,也就是這個原始參數的值,這個跟之前所談的傳值是一樣的。如果在函數中改變了副本的 值不會改變原始的值.   如果參數類型是引用類型,那麼傳過來的就是這個引用參數的副本,這個副本存放的是參數的地址。如果在函數中沒有改變這個副本的地址,而是改變了地址中的 值,那麼在函數內的改變會影響到傳入的參數。如果在函數中改變了副本的地址,如new一個,那麼副本就指向了一個新的地址,此時傳入的參數還是指向原來的 地址,所以不會改變參數的值。   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:說明

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

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

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved