溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Java深度復(fù)制功能與用法實例分析

發(fā)布時間:2020-08-23 19:08:35 來源:腳本之家 閱讀:105 作者:PursueCloud 欄目:編程語言

本文實例講述了Java深度復(fù)制功能與用法。分享給大家供大家參考,具體如下:

寫在前面:

什么是深度復(fù)制?在Java里面,在創(chuàng)建一個對象,我們通常會有一個引用指向該對象,當我們通過引用變量改變對象的值(屬性)時,引用是不變的,變的是內(nèi)存里面的那塊內(nèi)存,即引用所指向的對象。一般情況下,我們將該引用賦給另一個引用變量或者作為參數(shù)傳遞時,傳遞的也只是引用,即將引用指向“復(fù)制”了一份給另一個引用變量,隨后該引用變量也指向同一個對象,內(nèi)存里面并沒有創(chuàng)建一個新的對象。在某些情況下,我們需要“真正復(fù)制”對象,創(chuàng)建一份已知對象的copy,而不僅僅“復(fù)制”引用,用作備份也好,其他操作也好。

那么,該如何實現(xiàn)?

先說下思路:首先將對象序列化到流里,然后再反序列化,從流里讀取出來即可。

下面上代碼:

package com.yo.java;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;
import java.util.ArrayList;
import java.util.List;
/**
 * Java 實現(xiàn)深度復(fù)制
 * @author Yo
 *
 */
public class DeepCopy implements Serializable{
  int i;
  /**
   * @param args
   * @throws IOException
   * @throws ClassNotFoundException
   */
  public static void main(String[] args) throws ClassNotFoundException, IOException {
    demo1();
    demo2();
  }
  /**
   * 深度復(fù)制,實參類必須實現(xiàn)Serializable接口
   * @param o
   * @return
   * @throws IOException
   * @throws ClassNotFoundException
   */
  public static Object deepCopy(Object o) throws IOException, ClassNotFoundException {
//   //先序列化,寫入到流里
    ByteArrayOutputStream bo = new ByteArrayOutputStream();
    ObjectOutputStream oo = new ObjectOutputStream(bo);
    oo.writeObject(o);
    //然后反序列化,從流里讀取出來,即完成復(fù)制
    ByteArrayInputStream bi = new ByteArrayInputStream(bo.toByteArray());
    ObjectInputStream oi = new ObjectInputStream(bi);
    return oi.readObject();
  }
  /**
   * 引用傳遞與深度復(fù)制
   * @throws ClassNotFoundException
   * @throws IOException
   */
  public static void demo1() throws ClassNotFoundException, IOException {
    System.out.println("===========未使用深度復(fù)制=========");
    DeepCopy dc1 = new DeepCopy();
    dc1.i = 1;//初始化dc1里i的值
    DeepCopy dc2 = dc1;
    dc1.i = 2;//改變dc1里i的值
    System.out.println("dc1 : " + dc1.i);
    System.out.println("dc2(引用傳遞) : " + dc2.i);
    System.out.println("===========使用深度復(fù)制=========");
    DeepCopy dc3 = new DeepCopy();
    dc3.i = 1;//初始化dc3里i的值
    DeepCopy dc4 = (DeepCopy)deepCopy(dc3);
    dc3.i = 2;//改變dc3里i的值
    System.out.println("dc3 : " + dc3.i);
    System.out.println("dc4(深度復(fù)制) : " + dc4.i);
  }
  /**
   * 集合的值復(fù)制與深度復(fù)制
   * @throws ClassNotFoundException
   * @throws IOException
   */
  public static void demo2() throws ClassNotFoundException, IOException {
    System.out.println("===========未使用深度復(fù)制=========");
    //創(chuàng)建DeepCopy對象,并初始化i的值,添加到list1集合
    DeepCopy dc = new DeepCopy();
    dc.i = 1;//初始化dc1里i的值
    List<DeepCopy> list1 = new ArrayList<DeepCopy>();
    list1.add(dc);
    //未使用深度復(fù)制
    List<DeepCopy> list2 = new ArrayList<DeepCopy>(list1);//這里與使用Collections.copy(dest, src)結(jié)果一樣
    //改變list1中元素的值
    for(DeepCopy d1 : list1) {
      //改變dc1里i的值
      d1.i = 2;
    }
    //遍歷list
    for(DeepCopy d1 : list1) {
      System.out.println("list1 : " + d1.i);
    }
    //遍歷list2
    for(DeepCopy d2 : list2) {
      System.out.println("list2(復(fù)制) : " + d2.i);
    }
    System.out.println("===========使用深度復(fù)制后=========");
    DeepCopy dc3 = new DeepCopy();
    dc3.i = 1;//初始化dc3里i的值
    List<DeepCopy> list3 = new ArrayList<DeepCopy>();
    list3.add(dc3);
    List<DeepCopy> list4 = (List<DeepCopy>) deepCopy(list3);
    for(DeepCopy d : list3) {
      //改變dc3里i的值
      d.i = 2;
    }
    for(DeepCopy d3 : list3) {
      System.out.println("list3 : " + d3.i);
    }
    for(DeepCopy d4 : list4) {
      System.out.println("list4(深度復(fù)制): " + d4.i);
    }
  }
}

以上運行結(jié)果如下(實測):

===========未使用深度復(fù)制=========
dc1 : 2
dc2(引用傳遞) : 2
===========使用深度復(fù)制=========
dc3 : 2
dc4(深度復(fù)制) : 1
===========未使用深度復(fù)制=========
list1 : 2
list2(復(fù)制) : 2
===========使用深度復(fù)制后=========
list3 : 2
list4(深度復(fù)制): 1

由此可見,當僅僅只是引用傳遞或者根據(jù)對象的值創(chuàng)建新的值,僅能稱為“淺復(fù)制”,當原對象的屬性發(fā)生改變時,根據(jù)上面方式創(chuàng)建的新對象的屬性也會隨之改變;而如果采用深度復(fù)制,那是真正的copy了一份新的對象,新對象的與原對象不存在任何關(guān)聯(lián),原對象的屬性發(fā)生改變不會影響新對象,就如同copy的意義一樣

如上,如有不妥,如能指出,非常感謝

更多java相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《Java數(shù)據(jù)結(jié)構(gòu)與算法教程》、《Java操作DOM節(jié)點技巧總結(jié)》、《Java文件與目錄操作技巧匯總》和《Java緩存操作技巧匯總》

希望本文所述對大家java程序設(shè)計有所幫助。

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI