天天看點

細說 Java 的深拷貝和淺拷貝

任何變成語言中,其實都有淺拷貝和深拷貝的概念,Java 中也不例外。在對一個現有的對象進行拷貝操作的時候,是有淺拷貝和深拷貝之分的,他們在實際使用中,差別很大,如果對其進行混淆,可能會引發一些難以排查的問題。

本文就在 Java 中的深拷貝和淺拷貝做一個詳細的解說。

首先需要明白,淺拷貝和深拷貝都是針對一個已有對象的操作。那先來看看淺拷貝和深拷貝的概念。

在 Java 中,除了基本資料類型(元類型)之外,還存在 類的執行個體對象 這個引用資料類型。而一般使用 『 = 』号做指派操作的時候。對于基本資料類型,實際上是拷貝的它的值,但是對于對象而言,其實指派的隻是這個對象的引用,将原對象的引用傳遞過去,他們實際上還是指向的同一個對象。

而淺拷貝和深拷貝就是在這個基礎之上做的區分,如果在拷貝這個對象的時候,隻對基本資料類型進行了拷貝,而對引用資料類型隻是進行了引用的傳遞,而沒有真實的建立一個新的對象,則認為是淺拷貝。反之,在對引用資料類型進行拷貝的時候,建立了一個新的對象,并且複制其内的成員變量,則認為是深拷貝。

是以到現在,就應該了解了,所謂的淺拷貝和深拷貝,隻是在拷貝對象的時候,對 類的執行個體對象 這種引用資料類型的不同操作而已。

總結來說:

1、淺拷貝:對基本資料類型進行值傳遞,對引用資料類型進行引用傳遞般的拷貝,此為淺拷貝。

2、深拷貝:對基本資料類型進行值傳遞,對引用資料類型,建立一個新的對象,并複制其内容,此為深拷貝。

在 Java 中,所有的 Class 都繼承自 Object ,而在 Object 上,存在一個 clone() 方法,它被聲明為了 <code>protected</code> ,是以我們可以在其子類中,使用它。

而無論是淺拷貝還是深拷貝,都需要實作 clone() 方法,來完成操作。

可以看到,它的實作非常的簡單,它限制所有調用 clone() 方法的對象,都必須實作 <code>Cloneable</code> 接口,否者将抛出 <code>CloneNotSupportedException</code> 這個異常。最終會調用 <code>internalClone()</code> 方法來完成具體的操作。而 <code>internalClone()</code> 方法,實則是一個 native 的方法。對此我們就沒必要深究了,隻需要知道它可以 <code>clone()</code> 一個對象得到一個新的對象執行個體即可。

而反觀 Cloneable 接口,可以看到它其實什麼方法都不需要實作。對他可以簡單的了解隻是一個标記,是開發者允許這個對象被拷貝。

先來看看淺拷貝的例子。

首先建立一個 class 為 FatherClass ,對其實作 Cloneable 接口,并且重寫 <code>clone()</code> 方法。

然後先正常 new 一個 FatherClass 對象,再使用 clone() 方法建立一個新的對象。

最後看看輸出的 Log :

可以看到,使用 <code>clone()</code> 方法,從 == 和 hashCode 的不同可以看出,<code>clone()</code> 方法實則是真的建立了一個新的對象。

但這隻是一次淺拷貝的操作。

來驗證這一點,繼續看下去,在 FatherClass 中,還有一個 ChildClass 的對象 child ,clone() 方法是否也可以正常複制它呢?改寫一個上面的 Demo。

看到,這裡将其内的 child 進行負責,用起來看看輸出的 Log 效果。

從最後對 child 的輸出可以看到,A 和 B 的 child 對象,實際上還是指向了統一個對象,隻對對它的引用進行了傳遞。

既然已經了解了對 clone() 方法,隻能對目前對象進行淺拷貝,引用類型依然是在傳遞引用。

那麼,如何進行一個深拷貝呢?

比較常用的方案有兩種:

序列化(serialization)這個對象,再反序列化回來,就可以得到這個新的對象,無非就是序列化的規則需要我們自己來寫。

繼續利用 clone() 方法,既然 clone() 方法,是我們來重寫的,實際上我們可以對其内的引用類型的變量,再進行一次 clone()。

繼續改寫上面的 Demo ,讓 ChildClass 也實作 Cloneable 接口。

最重要的代碼就在 FatherClass.clone() 中,它對其内的 child ,再進行了一次 clone() 操作。

再來看看輸出的 Log。

可以看到,對 child 也進行了一次拷貝,這實則是對 ChildClass 進行的淺拷貝,但是對于 FatherClass 而言,則是一次深拷貝。

其實深拷貝的思路都差不多,序列化也好,使用 clone() 也好,實際上都是需要我們自己來編寫拷貝的規則,最終實作深拷貝的目的。

如果想要實作深拷貝,推薦使用 clone() 方法,這樣隻需要每個類自己維護自己即可,而無需關心内部其他的對象中,其他的參數是否也需要 clone() 。

到現在基本上就已經梳理清楚,Java 中淺拷貝和深拷貝的概念了。

實則淺拷貝和深拷貝隻是相對的,如果一個對象内部隻有基本資料類型,那用 clone() 方法擷取到的就是這個對象的深拷貝,而如果其内部還有引用資料類型,那用 clone() 方法就是一次淺拷貝的操作。

本文轉自承香墨影部落格園部落格,原文連結:http://www.cnblogs.com/plokmju/p/7357205.html,如需轉載請自行聯系原作者