淺析Java中的深拷貝與淺拷貝
首先我們看看淺拷貝和深拷貝的定義
淺拷貝:只復(fù)制一個(gè)對(duì)象,對(duì)象內(nèi)部存在的指向其他對(duì)象數(shù)組或者引用則不復(fù)制
深拷貝:對(duì)象,對(duì)象內(nèi)部的引用均復(fù)制
為了更好的理解它們的區(qū)別我們假設(shè)有一個(gè)對(duì)象A,它包含有2對(duì)象對(duì)象A1和對(duì)象A2
對(duì)象A進(jìn)行淺拷貝后,得到對(duì)象B但是對(duì)象A1和A2并沒有被拷貝
對(duì)象A進(jìn)行深拷貝,得到對(duì)象B的同時(shí)A1和A2連同它們的引用也被拷貝
在理解了深拷貝和淺拷貝后,我們來看看Java的深拷貝和淺拷貝實(shí)現(xiàn)。java.lang.Object的clone()方法默認(rèn)是返回一個(gè)前拷貝對(duì)象。因此如果要用clone()方法實(shí)現(xiàn)一個(gè)深拷貝,我們必須對(duì)每個(gè)對(duì)象的clone()方法進(jìn)行特別實(shí)現(xiàn)。當(dāng)對(duì)象層次復(fù)雜的時(shí)候,這樣做不但困難而且浪費(fèi)時(shí)間和容易出現(xiàn)錯(cuò)誤,特別有時(shí)候你不但需要深拷貝同時(shí)你也對(duì)這個(gè)對(duì)象進(jìn)行淺拷貝的時(shí)候,你會(huì)發(fā)現(xiàn)寫這個(gè)clone()方法真不是一個(gè)好的解決方案。
那么除了clone()方法,我們還可以怎么實(shí)現(xiàn)呢?答案是序列化,實(shí)現(xiàn)步驟和思路是把要拷貝的對(duì)象輸出成byte array,然后再利用ObjectInputStream轉(zhuǎn)換出新的對(duì)象。下面是代碼
public static Object copy(Object oldObj) { Object obj = null; try { // Write the object out to a byte array ByteArrayOutputStream bos = new ByteArrayOutputStream(); ObjectOutputStream out = new ObjectOutputStream(bos); out.writeObject(oldObj); out.flush(); out.close(); // Retrieve an input stream from the byte array and read // a copy of the object back in. ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray()); ObjectInputStream in = new ObjectInputStream(bis); obj = in.readObject(); } catch (IOException e) { e.printStackTrace(); } catch (ClassNotFoundException cnfe) { cnfe.printStackTrace(); } return obj; } |
【編輯推薦】