1. 程式人生 > >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:說明

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

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