posts - 89,  comments - 98,  trackbacks - 0
          關于對值傳遞與引用傳遞我一直似是而非。

          今天有個朋友問起我這個問題的時候我還是很困惑。

          對值傳遞我還是用一個例子來說明一下吧:

          /* 例 1 */
          /**
          * @(#) Test.java
          * @author fancy
          */
          public class Test {
          public static void test(boolean test) {
          test = ! test;
          System.out.println("In test(boolean) : test = " + test);
          }
          public static void main(String[] args) {
          boolean test = true;
          System.out.println("Before test(boolean) : test = " + test);
          test(test);
          System.out.println("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 變量沒有影響。那說明,參數類型是簡單類型的時候,是按值傳遞的。以參數形式傳遞簡單類型的變量時,實際上是將參數的值作了一個拷貝傳進方法函數的,那么在方法函數里再怎么改變其值,其結果都是只改變了拷貝的值,而不是源值。

          Java 是傳值還是傳引用,問題主要出在對象的傳遞上,因為 Java 中簡單類型沒有引用。既然爭論中提到了引用這個東西,為了搞清楚這個問題,我們必須要知道引用是什么。
            簡單的說,引用其實就像是一個對象的名字或者別名 (alias),一個對象在內存中會請求一塊空間來保存數據,根據對象的大小,它可能需要占用的空間大小也不等。訪問對象的時候,我們不會直接是訪問對象在內存中的數據,而是通過引用去訪問。引用也是一種數據類型,我們可以把它想象為類似 C 語言中指針的東西,它指示了對象在內存中的地址——只不過我們不能夠觀察到這個地址究竟是什么。
            如果我們定義了不止一個引用指向同一個對象,那么這些引用是不相同的,因為引用也是一種數據類型,需要一定的內存空間來保存。但是它們的值是相同的,都指示同一個對象在內存的中位置。比如
          String a = "Hello";
          String b = a;
            這里,a 和 b 是不同的兩個引用,我們使用了兩個定義語句來定義它們。但它們的值是一樣的,都指向同一個對象 "Hello"。也許你還覺得不夠直觀,因為 String 對象的值本身是不可更改的 (像 b = "World"; b = a; 這種情況不是改變了 "World" 這一對象的值,而是改變了它的引用 b 的值使之指向了另一個 String 對象 a)。那么我們用 StringBuffer 來舉一個例子:
          /* 例 2 */
          /**
          * @(#) Test.java
          * @author fancy
          */
          public class Test {
          public static void main(String[] args) {
          StringBuffer a = new StringBuffer("Hello");
          StringBuffer b = a;
          b.append(", World");
          System.out.println("a is " + a);
          }
          }
            運行結果:
          a is Hello, World
            這個例子中 a 和 b 都是引用,當改變了 b 指示的對象的值的時候,從輸出結果來看,a 所指示的對象的值也改變了。所以,a 和 b 都指向同一個對象即包含 "Hello" 的一個 StringBuffer 對象。
            這里我描述了兩個要點:
          1. 引用是一種數據類型,保存了對象在內存中的地址,這種類型即不是我們平時所說的簡單數據類型也不是類實例(對象);
          2. 不同的引用可能指向同一個對象,換句話說,一個對象可以有多個引用,即該類類型的變量。
          3. 對象是如何傳遞的呢
            關于對象的傳遞,有兩種說法,即“它是按值傳遞的”和“它是按引用傳遞的”。這兩種說法各有各的道理,但是它們都沒有從本質上去分析,即致于產生了爭論。   既然現在我們已經知道了引用是什么東西,那么現在不妨來分析一下對象作是參數是如何傳遞的。還是先以一個程序為例:
          /* 例 3 */
          /**
          * @(#) Test.java
          * @author fancy
          */
          public class Test {
          public static void test(StringBuffer str) {
          str.append(", World!");
          }
          public static void main(String[] args) {
          StringBuffer string = new StringBuffer("Hello");
          test(string);
          System.out.println(string);
          }
          }
            運行結果:
          Hello, World!
            test(string) 調用了 test(StringBuffer) 方法,并將 string 作為參數傳遞了進去。這里 string 是一個引用,這一點是勿庸置疑的。前面提到,引用是一種數據類型,而且不是對象,所以它不可能按引用傳遞,所以它是按值傳遞的,它么它的值究竟是什么呢?是對象的地址。
            由此可見,對象作為參數的時候是按值傳遞的,對嗎?錯!為什么錯,讓我們看另一個例子:
          /* 例 4 */
          /**
          * @(#) Test.java
          * @author fancy
          */
          public class Test {
          public static void test(String str) {
          str = "World";
          }
          public static void main(String[] args) {
          String string = "Hello";
          test(string);
          System.out.println(string);
          }
          }
            運行結果:
          Hello
            為什么會這樣呢?因為參數 str 是一個引用,而且它與 string 是不同的引用,雖然它們都是同一個對象的引用。str = "World" 則改變了 str 的值,使之指向了另一個對象,然而 str 指向的對象改變了,但它并沒有對 "Hello" 造成任何影響,而且由于 string 和 str 是不同的引用,str 的改變也沒有對 string 造成任何影響,結果就如例中所示。
            其結果是推翻了參數按值傳遞的說法。那么,對象作為參數的時候是按引用傳遞的了?也錯!因為上一個例子的確能夠說明它是按值傳遞的。
            結果,就像光到底是波還是粒子的問題一樣,Java 方法的參數是按什么傳遞的問題,其答案就只能是:即是按值傳遞也是按引用傳遞,只是參照物不同,結果也就不同。
          posted on 2006-07-21 15:46 水煮三國 閱讀(289) 評論(0)  編輯  收藏 所屬分類: J2EE
          <2025年5月>
          27282930123
          45678910
          11121314151617
          18192021222324
          25262728293031
          1234567

          常用鏈接

          留言簿(4)

          隨筆分類(85)

          隨筆檔案(89)

          文章分類(14)

          文章檔案(42)

          收藏夾(37)

          java

          oracle

          Sybase

          搜索

          •  

          積分與排名

          • 積分 - 210994
          • 排名 - 266

          最新評論

          閱讀排行榜

          評論排行榜

          主站蜘蛛池模板: 西盟| 贵南县| 嘉祥县| 博白县| 大城县| 中山市| 自贡市| 昭觉县| 西昌市| 崇州市| 沂南县| 郧西县| 凤台县| 乌拉特后旗| 洛川县| 邹平县| 林口县| 武威市| 理塘县| 滦平县| 兴仁县| 桐城市| 洪雅县| 紫云| 三门县| 左贡县| 湖口县| 沙坪坝区| 潞城市| 晋江市| 高要市| 平凉市| 共和县| 广河县| 甘德县| 苍溪县| 瑞昌市| 商都县| 南和县| 当雄县| 南丹县|