• <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淺/深復制

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

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

            Java的clone()方法

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

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

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

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

            ⑵Java中對象的克隆

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

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

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

            ④在派生類中實現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()識別出你要復制的是哪一              // 個對象。               }               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);//修改學生2后,不影響學生1的值。         }     }
            運行結果:
            C:\java>java Student
            name=zhangsan,age=18

            說明:

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

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

                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;//學生1和學生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);//學生1的教授成為lisi,age為30。      } }
            運行結果:
            C:\java>java Student
            name=lisi,age=30

            那應該如何實現深層次的克隆,即修改s2的教授不會影響s1的教授,代碼改進如下。

            改進使學生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);//學生1的教授不改變。      } }   

            3.利用串行化來做深復制

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

            在Java語言里深復制一個對象,常常可以先使對象實現Serializable接口,然后把對象(實際上只是對象的一個拷貝)寫到一個流里(腌成咸菜),再從流里讀出來(把咸菜回鮮),便可以重建對象。

            如下為深復制源代碼。

            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()); 
            }

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

            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;//學生1和學生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); //學生1的教授不改變。      }   }

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

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

            導航

            <2025年5月>
            27282930123
            45678910
            11121314151617
            18192021222324
            25262728293031
            1234567

            統計

            常用鏈接

            留言簿

            隨筆分類

            隨筆檔案

            文章分類

            文章檔案

            my

            搜索

            最新評論

            閱讀排行榜

            評論排行榜

            伊人久久大香线蕉成人| 国产日韩久久久精品影院首页| 久久婷婷色香五月综合激情| 亚洲国产精品综合久久一线| 亚洲级αV无码毛片久久精品| 久久精品蜜芽亚洲国产AV| 国产精品久久久久乳精品爆| 无码国内精品久久综合88| 亚洲伊人久久大香线蕉苏妲己| 女人高潮久久久叫人喷水| 国产成人久久AV免费| 香蕉久久夜色精品国产2020| 久久国产欧美日韩精品| 亚洲欧美精品一区久久中文字幕 | 亚洲精品国产第一综合99久久| 色欲久久久天天天综合网 | 色综合久久88色综合天天| 亚洲精品tv久久久久久久久久| 成人免费网站久久久| 亚洲香蕉网久久综合影视 | 欧美日韩精品久久久免费观看| 国内精品人妻无码久久久影院| 人妻无码精品久久亚瑟影视 | 亚洲国产成人久久精品99| 国产精品久久永久免费| 久久精品亚洲AV久久久无码| 日本久久久久久久久久| 国产亚洲婷婷香蕉久久精品| 久久中文字幕人妻熟av女| avtt天堂网久久精品| 一本色道久久88精品综合 | 久久综合五月丁香久久激情| 青青青国产成人久久111网站| 久久精品天天中文字幕人妻| 亚洲AV无码久久| 久久国产精品无码一区二区三区| 亚洲AV日韩精品久久久久| 无码AV中文字幕久久专区| 精品国产乱码久久久久久郑州公司 | 蜜桃麻豆www久久国产精品| 久久强奷乱码老熟女网站|