按值傳遞仍是按引用傳遞

發佈於:2008年10月22日 21:25html

原文連接:http://column.ibeifeng.com/chenchen/20081022186.shtml3d

這個在Java裏面是常常被提起的問題,也有一些爭論,彷佛最後還有一個所謂的結論:「在Java裏面參數傳遞都是按值傳遞」。事實上,這很容易讓人迷惑,下面先分別看看什麼是按值傳遞,什麼是按引用傳遞,只要能正確理解,至於稱做按什麼傳遞就不是個大問題了。code

 

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===3htm

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

public class TempTest {
private void test1(A a){
blog

}
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:按引用傳遞的重要特色
傳遞的是值的引用,也就是說傳遞前和傳遞後都指向同一個引用(也就是同一個內存空間)。
示例以下:

第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的實例,內存分配示意以下:class

(2):運行第9行,是修改A實例裏面的age的值,運行後內存分配示意以下:test

(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私塾」;

相關文章
相關標籤/搜索