<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基礎
    主站蜘蛛池模板: 台湾一级毛片永久免费| 玖玖在线免费视频| 亚洲日本天堂在线| 亚洲AV无码AV吞精久久| 高潮毛片无遮挡高清免费视频| 高清免费久久午夜精品| 久久青青草原国产精品免费| 国产亚洲精品无码专区| 久久亚洲AV无码精品色午夜| 亚洲av中文无码乱人伦在线观看 | 亚洲色图视频在线观看| 久久九九全国免费| 亚洲成人在线电影| 亚洲熟女综合一区二区三区| 免费在线看黄的网站| 久久亚洲免费视频| 午夜视频免费在线观看| 国产午夜鲁丝片AV无码免费| 亚洲av永久无码精品漫画| 国产亚洲欧美日韩亚洲中文色| 99精品视频免费在线观看| 亚洲国产成人VA在线观看| 亚洲三级在线视频| 成人网站免费大全日韩国产| 亚洲第一成年免费网站| 亚洲精品无码国产| 国产成人综合亚洲绿色| 黄色片在线免费观看| 亚洲第一成年网站视频| 亚洲а∨天堂久久精品| 三年在线观看免费观看完整版中文| 亚洲AV美女一区二区三区| 久久受www免费人成_看片中文| 丰满亚洲大尺度无码无码专线 | 久99精品视频在线观看婷亚洲片国产一区一级在线 | 久久精品国产精品亚洲色婷婷| 99久在线国内在线播放免费观看| 亚洲成a人片在线看| 中文字幕视频免费| 久久青青草原亚洲AV无码麻豆| 国产91色综合久久免费|