月亮的太陽

          小乖的BLOG
          posts - 114, comments - 41, trackbacks - 0, articles - 27
            BlogJava :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理

          JAVA中參數是傳址的

          Posted on 2005-12-11 16:06 月亮的太陽 閱讀(912) 評論(0)  編輯  收藏 所屬分類: 編程
          在不同的java新聞組中,參數是傳值還是傳址一直是一個經常被爭辯的話題。誤解的中心是以下兩個事實:

          對象是傳引用的
          參數是傳值的
          這兩個能夠同時成立嗎?一個字:是!在java中,你從來沒有傳遞對象,你傳遞的僅僅是對象的引用!一句話,java是傳引用的。然而,當你傳遞一個參數,那么只有一種參數傳遞機制:傳值!

          通常,當程序員討論傳值和傳引用時,他們是指語言的參數傳遞機制,c++同時支持這兩種機制,因此,以前使用過c++的程序員開始好像不能確定的java是如何傳參數的。java語言為了事情變得簡單只支持參數傳值的機制。

          java中的變量有兩種類型:引用類型和原始類型。當他們被作為參數傳遞給方法時,他們都是傳值的。這是一個非常重要的差別,下面的代碼范例將說明這一點。在繼續前,我們有必要定義一下傳值和傳引用。

          傳值意味著當參數被傳遞給一個方法或者函數時,方法或者函數接收到的是原始值的副本。因此,如果方法或者函數修改了參數,受影響的只是副本,原始值保持不變。

          關于java中的參數傳遞的混亂是因為很多java程序員是從c++轉變過來的。c++有引用和非引用類型的變量,并且分別是通過傳引用和傳值得。java語言有原始類型和對象引用,那么,按照邏輯,java對于原始類型使用傳值而對引用是傳引用的,就像c++一樣。畢竟,你會想到如果你正在傳遞一個引用,那么它一定是傳引用的。這是一個很誘惑人的想法,但是是錯誤的!

          在c++和java中,當函數的參數不是引用時,你傳遞的是值得副本(傳值)。但是對于引用類型就不同了。在c++中,當參數是引用類型,你傳遞的是引用或者內存地址(傳引用),而在java中,傳遞一個引用類型的參數的結果只是傳遞引用的副本(傳值)而非引用自身。

          這是一個非常重要的區別!java不考慮參數的類型,一律傳遞參數的副本。仍然不信?如果java中是傳引用,那么下面的范例中的swap方法將交換他們的參數。因為是傳值,因此這個方法不是像期望的那樣正常工作。

          class Swap
          {
          public static void main(String args[])
          {
          Integer a, b;
          int i,j;
          a = new Integer(10);
          b = new Integer(50);
          i = 5;
          j = 9;
          System.out.println(\"Before Swap, a is \" + a);
          System.out.println(\"Before Swap, b is \" + b);
          swap(a, b);
          System.out.println(\"After Swap a is \" + a);
          System.out.println(\"After Swap b is \" + b);
          System.out.println(\"Before Swap i is \" + i);
          System.out.println(\"Before Swap j is \" + j);
          swap(i,j);
          System.out.println(\"After Swap i is \" + i);
          System.out.println(\"After Swap j is \" + j);
          }
          public static void swap(Integer ia, Integer ib)
          {
          Integer temp = ia;
          ia = ib;
          ib = temp;
          }
          public static void swap(int li, int lj)
          {
          int temp = li;
          li = lj;
          lj = temp;
          }
          }

          上面程序的輸出是:

          Before Swap, a is 10
          Before Swap, b is 50
          After Swap a is 10
          After Swap b is 50
          Before Swap i is 5
          Before Swap j is 9
          After Swap i is 5
          After Swap j is 9

          因為swap方法接收到的是引用參數的副本(傳值),對他們的修改不會反射到調用代碼。

          譯者注:在傳遞引用和原始類型時還是有不同的,考慮以下的代碼:

          class Change
          {
          public static void main(String args[])
          {
          StringBuffer a=new StringBuffer(\"ok\");
          int i;
          i = 5;
          System.out.println(\"Before change, a is \" + a);
          change(a);
          System.out.println(\"After change a is \" + a);
          System.out.println(\"Before change i is \" + i);
          change(i);
          System.out.println(\"After change i is \" + i);
          }
          public static void change(StringBuffer ia)
          {
          ia.append(\" ok?\");
          }
          public static void change(int li)
          {
          li = 10;
          }
          }

          程序的輸出為:

          Before change, a is ok

          After change a is ok ok?
          Before change i is 5
          After change i is 5

          即如果傳遞的是引用,那么可以修改引用對象的內容,這個改變會影響到原來的對象,而傳遞的如果是原始類型則不會有影響。這個也是造成誤解的原因之一吧。
          海邊的貝殼
          主站蜘蛛池模板: 石家庄市| 桃园市| 洱源县| 类乌齐县| 和平区| 宁津县| 阳泉市| 蒙城县| 贵州省| 柯坪县| 平山县| 进贤县| 上犹县| 温州市| 安丘市| 舞钢市| 公主岭市| 安图县| 定边县| 县级市| 宝清县| 府谷县| 永修县| 天津市| 卢氏县| 大埔区| 舞阳县| 子洲县| 长子县| 永城市| 叶城县| 五常市| 平和县| 湟源县| 肃北| 林甸县| 新民市| 苗栗县| 宕昌县| 涞源县| 赤城县|