java中值傳遞和引用傳遞


java函數的值傳遞和引用傳遞,在實際工作中遇到的時候總是有點模糊,故寫這篇文章加深對java函數的理解

       

1. 值傳遞:

在方法調用時候,傳遞的參數是按值的拷貝傳遞(一般是基礎數據類型,int,short,byte,boolean,float,double,char,long)。示例如下:

/**
 * Created by panther on 17-3-21.
 */
public class Main {


    private void test1(int a) {
        System.out.println(a);
    }

    public static void main(String[] args) {
        Main t = new Main();
        int a = 3;
        t.test1(a);//這裏傳遞的參數a就是按值傳遞
    }
}


 

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

示例如下:

/**
 * Created by panther on 17-3-21.
 */
public class Main {


    private void test1(int a) {
        a = 5;
        System.out.println("test1方法中的a=" + a);
    }

    public static void main(String[] args) {
        Main t = new Main();
        int a = 3;
        t.test1(a);//這裏傳遞的參數a就是按值傳遞
        System.out.println("main方法中的a=" + a);
    }
}


運行結果是:



2. 按引用傳遞

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

示例如下:

/**
 * Created by panther on 17-3-21.
 */
public class Main {


    private void test1(A a) {
        a.age = 10;
        System.out.println("test1方法中的a=" + a.age);
    }

    public static void main(String[] args) {
        Main t = new Main();
        A a = new A();
        a.age = 3;
        t.test1(a);//這裏傳遞的參數a1就是按引用傳遞
        System.out.println("main方法中的a=" + a.age);
    }


    static class A{
        public int age = 0;
    }
}

運行結果:

 

上面結果表明:

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


3:理解按引用傳遞的過程——內存分配示意圖

要想正確理解按引用傳遞的過程,就必須學會理解內存分配的過程,內存分配示意圖可以輔助我們去理解這個過程。

用上面的例子來進行分析:

(1):運行開始,運行第14行,創建了一個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


4:對上述例子的改變

理解了上面的例子,可能有人會問,那麼能不能讓按照引用傳遞的值,相互不影響呢?就是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. }  

運行結果爲:

  1. test1方法中的age=20  
  2. main方法中的age=10  

爲什麼這次的運行結果和前面的例子不一樣呢,還是使用內存示意圖來理解一下

5:再次理解按引用傳遞

(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

6:說明

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

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

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章