posts - 89,  comments - 98,  trackbacks - 0
          關(guān)于對(duì)值傳遞與引用傳遞我一直似是而非。

          今天有個(gè)朋友問(wèn)起我這個(gè)問(wèn)題的時(shí)候我還是很困惑。

          對(duì)值傳遞我還是用一個(gè)例子來(lái)說(shuō)明一下吧:

          /* 例 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);
          }
          }
            運(yùn)行結(jié)果:
          Before test(boolean) : test = true
          In test(boolean) : test = false
          After test(boolean) : test = true
            不難看出,雖然在 test(boolean) 方法中改變了傳進(jìn)來(lái)的參數(shù)的值,但對(duì)這個(gè)參數(shù)源變量本身并沒(méi)有影響,即對(duì) main(String[]) 方法里的 test 變量沒(méi)有影響。那說(shuō)明,參數(shù)類型是簡(jiǎn)單類型的時(shí)候,是按值傳遞的。以參數(shù)形式傳遞簡(jiǎn)單類型的變量時(shí),實(shí)際上是將參數(shù)的值作了一個(gè)拷貝傳進(jìn)方法函數(shù)的,那么在方法函數(shù)里再怎么改變其值,其結(jié)果都是只改變了拷貝的值,而不是源值。

          Java 是傳值還是傳引用,問(wèn)題主要出在對(duì)象的傳遞上,因?yàn)?Java 中簡(jiǎn)單類型沒(méi)有引用。既然爭(zhēng)論中提到了引用這個(gè)東西,為了搞清楚這個(gè)問(wèn)題,我們必須要知道引用是什么。
            簡(jiǎn)單的說(shuō),引用其實(shí)就像是一個(gè)對(duì)象的名字或者別名 (alias),一個(gè)對(duì)象在內(nèi)存中會(huì)請(qǐng)求一塊空間來(lái)保存數(shù)據(jù),根據(jù)對(duì)象的大小,它可能需要占用的空間大小也不等。訪問(wèn)對(duì)象的時(shí)候,我們不會(huì)直接是訪問(wèn)對(duì)象在內(nèi)存中的數(shù)據(jù),而是通過(guò)引用去訪問(wèn)。引用也是一種數(shù)據(jù)類型,我們可以把它想象為類似 C 語(yǔ)言中指針的東西,它指示了對(duì)象在內(nèi)存中的地址——只不過(guò)我們不能夠觀察到這個(gè)地址究竟是什么。
            如果我們定義了不止一個(gè)引用指向同一個(gè)對(duì)象,那么這些引用是不相同的,因?yàn)橐靡彩且环N數(shù)據(jù)類型,需要一定的內(nèi)存空間來(lái)保存。但是它們的值是相同的,都指示同一個(gè)對(duì)象在內(nèi)存的中位置。比如
          String a = "Hello";
          String b = a;
            這里,a 和 b 是不同的兩個(gè)引用,我們使用了兩個(gè)定義語(yǔ)句來(lái)定義它們。但它們的值是一樣的,都指向同一個(gè)對(duì)象 "Hello"。也許你還覺得不夠直觀,因?yàn)?String 對(duì)象的值本身是不可更改的 (像 b = "World"; b = a; 這種情況不是改變了 "World" 這一對(duì)象的值,而是改變了它的引用 b 的值使之指向了另一個(gè) String 對(duì)象 a)。那么我們用 StringBuffer 來(lái)舉一個(gè)例子:
          /* 例 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);
          }
          }
            運(yùn)行結(jié)果:
          a is Hello, World
            這個(gè)例子中 a 和 b 都是引用,當(dāng)改變了 b 指示的對(duì)象的值的時(shí)候,從輸出結(jié)果來(lái)看,a 所指示的對(duì)象的值也改變了。所以,a 和 b 都指向同一個(gè)對(duì)象即包含 "Hello" 的一個(gè) StringBuffer 對(duì)象。
            這里我描述了兩個(gè)要點(diǎn):
          1. 引用是一種數(shù)據(jù)類型,保存了對(duì)象在內(nèi)存中的地址,這種類型即不是我們平時(shí)所說(shuō)的簡(jiǎn)單數(shù)據(jù)類型也不是類實(shí)例(對(duì)象);
          2. 不同的引用可能指向同一個(gè)對(duì)象,換句話說(shuō),一個(gè)對(duì)象可以有多個(gè)引用,即該類類型的變量。
          3. 對(duì)象是如何傳遞的呢
            關(guān)于對(duì)象的傳遞,有兩種說(shuō)法,即“它是按值傳遞的”和“它是按引用傳遞的”。這兩種說(shuō)法各有各的道理,但是它們都沒(méi)有從本質(zhì)上去分析,即致于產(chǎn)生了爭(zhēng)論。   既然現(xiàn)在我們已經(jīng)知道了引用是什么東西,那么現(xiàn)在不妨來(lái)分析一下對(duì)象作是參數(shù)是如何傳遞的。還是先以一個(gè)程序?yàn)槔?br />/* 例 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);
          }
          }
            運(yùn)行結(jié)果:
          Hello, World!
            test(string) 調(diào)用了 test(StringBuffer) 方法,并將 string 作為參數(shù)傳遞了進(jìn)去。這里 string 是一個(gè)引用,這一點(diǎn)是勿庸置疑的。前面提到,引用是一種數(shù)據(jù)類型,而且不是對(duì)象,所以它不可能按引用傳遞,所以它是按值傳遞的,它么它的值究竟是什么呢?是對(duì)象的地址。
            由此可見,對(duì)象作為參數(shù)的時(shí)候是按值傳遞的,對(duì)嗎?錯(cuò)!為什么錯(cuò),讓我們看另一個(gè)例子:
          /* 例 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);
          }
          }
            運(yùn)行結(jié)果:
          Hello
            為什么會(huì)這樣呢?因?yàn)閰?shù) str 是一個(gè)引用,而且它與 string 是不同的引用,雖然它們都是同一個(gè)對(duì)象的引用。str = "World" 則改變了 str 的值,使之指向了另一個(gè)對(duì)象,然而 str 指向的對(duì)象改變了,但它并沒(méi)有對(duì) "Hello" 造成任何影響,而且由于 string 和 str 是不同的引用,str 的改變也沒(méi)有對(duì) string 造成任何影響,結(jié)果就如例中所示。
            其結(jié)果是推翻了參數(shù)按值傳遞的說(shuō)法。那么,對(duì)象作為參數(shù)的時(shí)候是按引用傳遞的了?也錯(cuò)!因?yàn)樯弦粋€(gè)例子的確能夠說(shuō)明它是按值傳遞的。
            結(jié)果,就像光到底是波還是粒子的問(wèn)題一樣,Java 方法的參數(shù)是按什么傳遞的問(wèn)題,其答案就只能是:即是按值傳遞也是按引用傳遞,只是參照物不同,結(jié)果也就不同。
          posted on 2006-07-21 15:46 水煮三國(guó) 閱讀(292) 評(píng)論(0)  編輯  收藏 所屬分類: J2EE

          只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。


          網(wǎng)站導(dǎo)航:
           
          <2025年7月>
          293012345
          6789101112
          13141516171819
          20212223242526
          272829303112
          3456789

          常用鏈接

          留言簿(4)

          隨筆分類(85)

          隨筆檔案(89)

          文章分類(14)

          文章檔案(42)

          收藏夾(37)

          java

          oracle

          Sybase

          搜索

          •  

          積分與排名

          • 積分 - 211656
          • 排名 - 266

          最新評(píng)論

          閱讀排行榜

          評(píng)論排行榜

          主站蜘蛛池模板: 建德市| 兴业县| 昌都县| 汾西县| 轮台县| 淮南市| 陵水| 雅江县| 通州区| 古丈县| 澄城县| 巴马| 盘山县| 德化县| 垦利县| 墨竹工卡县| 安龙县| 湛江市| 临颍县| 虞城县| 瑞金市| 新兴县| 嘉禾县| 新沂市| 天祝| 溆浦县| 阿尔山市| 昌吉市| 广宁县| 怀来县| 新田县| 南投县| 庆云县| 景洪市| 二连浩特市| 安宁市| 灵台县| 宜城市| 随州市| 江门市| 五华县|