Java到底是傳值還是傳引用?相信很少有人能完全回答正確。通常的說法是:對於基本資料型別(整型、浮點型、字元型、布林型等),傳值;對於引用型別(物件、陣列),傳引用。基本型別傳值,所有人都不會對此有疑義;問題出在引用型別上。下面是小編為大家搜尋整理的java傳值的方法,希望能給大家帶來幫助!更多精彩內容請及時關注我們應屆畢業生考試網!
1. 簡單型別是按值傳遞的
Java 方法的引數是簡單型別的時候,是按值傳遞的 (pass by value)。這一點我們可以通過一個簡單的例子來說明:
/* 例 1 */
/**
* @(#)
* @author fancy
*/
public class Test {
public static void test(boolean test) {
test = ! test;
tln("In test(boolean) : test = " + test);
}
public static void main(String[] args) {
boolean test = true;
tln("Before test(boolean) : test = " + test);
test(test);
tln("After test(boolean) : test = " + test);
}
}
執行結果:
Before test(boolean) : test = true
In test(boolean) : test = false
After test(boolean) : test = true
不難看出,雖然在 test(boolean) 方法中改變了傳進來的引數的值,但對這個引數源變數本身並沒有影響,即對 main(String[]) 方法裡的 test 變數沒有影響。那說明,引數型別是簡單型別的時候,是按值傳遞的。以引數形式傳遞簡單型別的變數時,實際上是將引數的值作了一個拷貝傳進方法函式的,那麼在方法函式裡再怎麼改變其值,其結果都是隻改變了拷貝的值,而不是源值。
2. 什麼是引用
Java 是傳值還是傳引用,問題主要出在物件的傳遞上,因為 Java 中簡單型別沒有引用。既然爭論中提到了引用這個東西,為了搞清楚這個問題,我們必須要知道引用是什麼。
簡單的說,引用其實就像是一個物件的名字或者別名 (alias),一個物件在記憶體中會請求一塊空間來儲存資料,根據物件的大小,它可能需要佔用的空間大小也不等。訪問物件的時候,我們不會直接是訪問物件在記憶體中的資料,而是通過引用去訪問。引用也是一種資料型別,我們可以把它想象為類似 C 語言中指標的東西,它指示了物件在記憶體中的地址——只不過我們不能夠觀察到這個地址究竟是什麼。
如果我們定義了不止一個引用指向同一個物件,那麼這些引用是不相同的,因為引用也是一種資料型別,需要一定的記憶體空間來儲存。但是它們的值是相同的,都指示同一個物件在記憶體的中位置。比如
String a = "Hello";
String b = a;
這裡,a 和 b 是不同的兩個引用,我們使用了兩個定義語句來定義它們。但它們的值是一樣的,都指向同一個物件 "Hello"。也許你還覺得不夠直觀,因為 String 物件的值本身是不可更改的 (像 b = "World"; b = a; 這種情況不是改變了 "World" 這一物件的值,而是改變了它的引用 b 的值使之指向了另一個 String 物件 a)。那麼我們用 StringBuffer 來舉一個例子:
/* 例 2 */
/**
* @(#)
* @author fancy
*/
public class Test {
public static void main(String[] args) {
StringBuffer a = new StringBuffer("Hello");
StringBuffer b = a;
nd(", World");
tln("a is " + a);
}
}
執行結果:
a is Hello, World
這個例子中 a 和 b 都是引用,當改變了 b 指示的物件的值的時候,從輸出結果來看,a 所指示的物件的值也改變了。所以,a 和 b 都指向同一個物件即包含 "Hello" 的一個 StringBuffer 物件。
這裡我描述了兩個要點:
引用是一種資料型別,儲存了物件在記憶體中的地址,這種型別即不是我們平時所說的'簡單資料型別也不是類例項(物件);
不同的引用可能指向同一個物件,換句話說,一個物件可以有多個引用,即該類型別的變數。
3. 物件是如何傳遞的呢
關於物件的傳遞,有兩種說法,即“它是按值傳遞的”和“它是按引用傳遞的”。這兩種說法各有各的道理,但是它們都沒有從本質上去分析,即致於產生了爭論。
既然現在我們已經知道了引用是什麼東西,那麼現在不妨來分析一下物件作是引數是如何傳遞的。還是先以一個程式為例:
/* 例 3 */
/**
* @(#)
* @author fancy
*/
public class Test {
public static void test(StringBuffer str) {
nd(", World!");
}
public static void main(String[] args) {
StringBuffer string = new StringBuffer("Hello");
test(string);
tln(string);
}
}
執行結果:
Hello, World!
test(string) 呼叫了 test(StringBuffer) 方法,並將 string 作為引數傳遞了進去。這裡 string 是一個引用,這一點是勿庸置疑的。前面提到,引用是一種資料型別,而且不是物件,所以它不可能按引用傳遞,所以它是按值傳遞的,它麼它的值究竟是什麼呢?是物件的地址。
由此可見,物件作為引數的時候是按值傳遞的,對嗎?錯!為什麼錯,讓我們看另一個例子:
/* 例 4 */
/**
* @(#)
* @author fancy
*/
public class Test {
public static void test(String str) {
str = "World";
}
public static void main(String[] args) {
String string = "Hello";
test(string);
tln(string);
}
}