• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            chenglong7997

            java淺/深復(fù)制

            ⑴淺復(fù)制(淺克隆)
                被復(fù)制對象的所有變量都含有與原來的對象相同的值,而所有的對其他對象的引用仍然指向原來的對象。換言之,淺復(fù)制僅僅復(fù)制所考慮的對象,而不復(fù)制它所引用的對象。

            ⑵深復(fù)制(深克隆)
               被復(fù)制對象的所有變量都含有與原來的對象相同的值,除去那些引用其他對象的變量。那些引用其他對象的變量將指向被復(fù)制過的新對象,而不再是原有的那些被引用的對象。換言之,深復(fù)制把要復(fù)制的對象所引用的對象都復(fù)制了一遍。

            Java的clone()方法

            ⑴clone方法將對象復(fù)制了一份并返回給調(diào)用者。一般而言,clone()方法滿足:

            ①對任何的對象x,都有x.clone() !=x//克隆對象與原對象不是同一個對象

            ②對任何的對象x,都有x.clone().getClass()= =x.getClass()//克隆對象與原對象的類型一樣

            ③如果對象x的equals()方法定義恰當(dāng),那么x.clone().equals(x)應(yīng)該成立。

            ⑵Java中對象的克隆

            ①為了獲取對象的一份拷貝,我們可以利用Object類的clone()方法。

            ②在派生類中覆蓋基類的clone()方法,并聲明為public。

            ③在派生類的clone()方法中,調(diào)用super.clone()。

            ④在派生類中實現(xiàn)Cloneable接口。

            請看如下代碼:
                class Student implements Cloneable        {            String name;            int age;             Student(String name,int age)            {                this.name=name;                this.age=age;            }           public Object clone()           {              Object o=null;              try              {              o=(Student)super.clone();//Object中的clone()識別出你要復(fù)制的是哪一              // 個對象。               }               catch(CloneNotSupportedException e)              {                  System.out.println(e.toString());              }              return o;          }            public static void main(String[] args)          {            Student s1=new Student("zhangsan",18);            Student s2=(Student)s1.clone();            s2.name="lisi";            s2.age=20;          System.out.println("name="+s1.name+","+"age="+s1.age);//修改學(xué)生2后,不影響學(xué)生1的值。         }     }
            運行結(jié)果:
            C:\java>java Student
            name=zhangsan,age=18

            說明:

            ①為什么我們在派生類中覆蓋Object的clone()方法時,一定要調(diào)用super.clone()呢?在運行時刻,Object中的 clone()識別出你要復(fù)制的是哪一個對象,然后為此對象分配空間,并進(jìn)行對象的復(fù)制,將原始對象的內(nèi)容一一復(fù)制到新對象的存儲空間中。

            ②繼承自java.lang.Object類的clone()方法是淺復(fù)制。以下代碼可以證明之。

                class Professor        {            String name;            int age;            Professor(String name,int age)            {                this.name=name;                this.age=age;            }       }        public class Student implements Cloneable     {         String name;//常量對象。         int age;         Professor p;//學(xué)生1和學(xué)生2的引用值都是一樣的。         Student(String name,int age,Professor p)         {             this.name=name;             this.age=age;             this.p=p;         }        public Object clone()         {             Student o=null;            try             {                o=(Student)super.clone();             }             catch(CloneNotSupportedException e)            {                System.out.println(e.toString());            }             return o;         }           public static void main(String[] args)         {           Professor p=new Professor("wangwu",50);           Student s1=new Student("zhangsan",18,p);           Student s2=(Student)s1.clone();            s2.p.name="lisi";            s2.p.age=30;      System.out.println("name="+s1.p.name+","+"age="+s1.p.age);//學(xué)生1的教授成為lisi,age為30。      } }
            運行結(jié)果:
            C:\java>java Student
            name=lisi,age=30

            那應(yīng)該如何實現(xiàn)深層次的克隆,即修改s2的教授不會影響s1的教授,代碼改進(jìn)如下。

            改進(jìn)使學(xué)生1的Professor不改變(深層次的克隆)

                class Professor implements Cloneable        {            String name;            int age;         Professor(String name,int age)            {                this.name=name;                this.age=age;            }             public Object clone()           {               Object o=null;               try              {                   o=super.clone();              }              catch(CloneNotSupportedException e)              {                  System.out.println(e.toString());             }             return o;         }     }     public  class Student implements Cloneable       {           String name;           int age;           Professor p;           Student(String name,int age,Professor p)          {              this.name=name;              this.age=age;              this.p=p;          }       public Object clone()          {               Student o=null;               try               {                 o=(Student)super.clone();             }             catch(CloneNotSupportedException e)             {                 System.out.println(e.toString());             }            o.p=(Professor)p.clone();            return o;         }         public static void main(String[] args)        {          Professor p=new Professor("wangwu",50);          Student s1=new Student("zhangsan",18,p);          Student s2=(Student)s1.clone();           s2.p.name="lisi";           s2.p.age=30;      System.out.println("name="+s1.p.name+","+"age="+s1.p.age);//學(xué)生1的教授不改變。      } }   

            3.利用串行化來做深復(fù)制

              把對象寫到流里的過程是串行化(Serilization)過程,但是在Java程序師圈子里又非常形象地稱為“冷凍”或者“腌咸菜 (picking)”過程;而把對象從流中讀出來的并行化(Deserialization)過程則叫做“解凍”或者“回鮮(depicking)”過程。應(yīng)當(dāng)指出的是,寫在流里的是對象的一個拷貝,而原對象仍然存在于JVM里面,因此“腌成咸菜”的只是對象的一個拷貝,Java咸菜還可以回鮮。

            在Java語言里深復(fù)制一個對象,常??梢韵仁箤ο髮崿F(xiàn)Serializable接口,然后把對象(實際上只是對象的一個拷貝)寫到一個流里(腌成咸菜),再從流里讀出來(把咸菜回鮮),便可以重建對象。

            如下為深復(fù)制源代碼。

            public Object deepClone() 

              //將對象寫到流里 
              ByteArrayOutputStream bo=new ByteArrayOutputStream(); 
              ObjectOutputStream oo=new ObjectOutputStream(bo); 
              oo.writeObject(this); 

              //從流里讀出來 
              ByteArrayInputStream bi=new ByteArrayInputStream(bo.toByteArray()); 
              ObjectInputStream oi=new ObjectInputStream(bi); 
              return(oi.readObject()); 
            }

            這樣做的前提是對象以及對象內(nèi)部所有引用到的對象都是可串行化的,否則,就需要仔細(xì)考察那些不可串行化的對象可否設(shè)成transient,從而將之排除在復(fù)制過程之外。上例代碼改進(jìn)如下。

            import java.io.*;         class Professor implements Serializable        {            String name;            int age;            Professor(String name,int age)            {                this.name=name;                this.age=age;            }        }        public  class Student implements Serializable       {           String name;//常量對象。           int age;           Professor p;//學(xué)生1和學(xué)生2的引用值都是一樣的。            Student(String name,int age,Professor p)         {             this.name=name;             this.age=age;             this.p=p;        }      public Object deepClone() throws IOException, OptionalDataException,ClassNotFoundException       {       //將對象寫到流里       ByteArrayOutputStream bo=new ByteArrayOutputStream();       ObjectOutputStream oo=new ObjectOutputStream(bo);       oo.writeObject(this);          //從流里讀出來       ByteArrayInputStream bi=new ByteArrayInputStream(bo.toByteArray());       ObjectInputStream oi=new ObjectInputStream(bi);       return(oi.readObject());       }                  public static void main(String[] args) throws IOException,ClassNotFoundException       {            Professor p=new Professor("wangwu",50);            Student s1=new Student("zhangsan",18,p);            Student s2=(Student)s1.deepClone();            s2.p.name="lisi";            s2.p.age=30;           System.out.println("name="+s1.p.name+","+"age="+s1.p.age); //學(xué)生1的教授不改變。      }   }

            運行結(jié)果:
            C:\java>java Student
            name=wangwu,age=50 

            posted on 2012-04-11 13:48 Snape 閱讀(210) 評論(0)  編輯 收藏 引用 所屬分類: Java

            導(dǎo)航

            <2025年6月>
            25262728293031
            1234567
            891011121314
            15161718192021
            22232425262728
            293012345

            統(tǒng)計

            常用鏈接

            留言簿

            隨筆分類

            隨筆檔案

            文章分類

            文章檔案

            my

            搜索

            最新評論

            閱讀排行榜

            評論排行榜

            久久久国产99久久国产一| 无码人妻精品一区二区三区久久 | 免费精品久久久久久中文字幕| 久久国产影院| 亚洲va久久久噜噜噜久久狠狠 | 国产99久久久国产精品~~牛| 久久av免费天堂小草播放| 亚洲精品无码久久久影院相关影片 | 狠狠人妻久久久久久综合蜜桃| 日日狠狠久久偷偷色综合免费 | 久久99精品久久久久久水蜜桃| 久久伊人精品一区二区三区| 嫩草影院久久99| 97久久婷婷五月综合色d啪蜜芽| 91久久香蕉国产熟女线看| 亚洲国产精品无码久久久不卡| 亚洲国产精品一区二区久久| 日韩乱码人妻无码中文字幕久久| 国产精品美女久久久网AV| 久久99精品久久久久子伦| 免费精品久久天干天干| 精品欧美一区二区三区久久久| 久久久久人妻一区精品色| 久久精品国产一区二区电影| 久久99国产亚洲高清观看首页| 日本人妻丰满熟妇久久久久久| 无码8090精品久久一区 | 国产69精品久久久久777| 中文字幕久久波多野结衣av| 国产一区二区久久久| 久久亚洲精品国产精品婷婷| 久久精品中文字幕一区| 国产精品久久久久久久午夜片 | 久久久久久久久久久| 亚洲国产香蕉人人爽成AV片久久 | 伊人久久综在合线亚洲2019| 国产V综合V亚洲欧美久久| 久久精品天天中文字幕人妻| 日日噜噜夜夜狠狠久久丁香五月 | 久久这里有精品视频| 久久人妻少妇嫩草AV无码蜜桃|