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私塾”;