<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    隨筆-59  評論-31  文章-0  trackbacks-0

    1.淺復制和深復制概念
    ⑴淺復制(淺克隆)
    被復制對象的任何變量都含有和原來的對象相同的值,而任何的對其他對象的引用仍然指向原來的對象。換言之,淺復制僅僅復制所考慮的對象,而不復制他所引用的對象。
     
    ⑵深復制(深克隆)
    被復制對象的任何變量都含有和原來的對象相同的值,除去那些引用其他對象的變量。那些引用其他對象的變量將指向被復制過的新對象,而不再是原有的那些被引用的對象。換言之,深復制把要復制的對象所引用的對象都復制了一遍。
     
    2.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的值。
       }
    }
     
     
    說明:
    ①為什么我們在派生類中覆蓋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;
        }
    }
    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());
            }
            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的教授成為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;
        }
    }
    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()
    {
     //將對象寫到流里
     ByteArrayOutoutStream 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,從而將之排除在復制過程之外。上例代碼改進如下。
     
    class Professor implements Serializable
    {
        String name;
        int age;
        Professor(String name,int age)
        {
            this.name=name;
            this.age=age;
        }
    }
    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
    {
     //將對象寫到流里
     ByteArrayOutoutStream 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)
        {
          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的教授不改變。
    }
    }

    posted on 2009-06-20 18:03 RoyPayne 閱讀(171) 評論(0)  編輯  收藏 所屬分類: java基礎
    主站蜘蛛池模板: 99亚洲精品卡2卡三卡4卡2卡| 免费视频淫片aa毛片| 美女视频免费看一区二区| 久久亚洲精品中文字幕| 亚洲日韩国产精品乱| 青青草国产免费久久久91 | heyzo亚洲精品日韩| 成人免费毛片视频| 四虎在线视频免费观看视频| 成人久久免费网站| 一级毛片大全免费播放| 亚洲av午夜电影在线观看 | 免费人成在线观看网站品爱网| 成年大片免费视频播放一级| 亚洲精品无码久久久久牙蜜区| 亚洲人成网男女大片在线播放| 亚洲成aⅴ人片在线观| 亚洲永久中文字幕在线| 久久久国产精品亚洲一区| 亚洲av无码乱码国产精品fc2| 狠狠亚洲狠狠欧洲2019| 国产日韩成人亚洲丁香婷婷| 亚洲高清无码专区视频| 免费一级特黄特色大片在线| 免费国产成人午夜电影| 国产精品无码免费视频二三区| 日韩在线免费播放| 最好免费观看韩国+日本| 成年女人色毛片免费看| 成人免费视频小说| 国内精品免费视频自在线| 日本免费一二区在线电影| 日韩一区二区免费视频| 免费网站看v片在线香蕉| 国产精品视_精品国产免费| 国产老女人精品免费视频| 亚洲成人影院在线观看| 国产亚洲精品成人a v小说| 国产亚洲av片在线观看播放| 久久精品九九亚洲精品天堂| 久久久久久亚洲AV无码专区|