重要接口—Cloneable接口
作用:允許實現了Cloneable接口的類的對象進行克隆操作。允許克隆的意思是可以調用clone()方法,至于是深拷貝還是淺拷貝,取決于如何重寫Object的clone()方法。查看Cloneable接口的源碼如下:
package java.lang; public interface Cloneable {}如果沒有實現Cloneable就調用clone()方法,會拋出異常。看下Object源碼就知道了:
protected Object clone() throws CloneNotSupportedException {if (!(this instanceof Cloneable)) {//這里會檢查是否是Cloneable的實例throw new CloneNotSupportedException("Class " + getClass().getName() +" doesn't implement Cloneable");}return internalClone(); }下面舉一個深拷貝的例子,ArrayList的clone()方法:
//深拷貝public Object clone() {try {ArrayList<?> v = (ArrayList<?>) super.clone();//新開辟一個內存空間給ArrayList的對象成員Object[] elementData;v.elementData = Arrays.copyOf(elementData, size);v.modCount = 0;return v;} catch (CloneNotSupportedException e) {// this shouldn't happen, since we are Cloneablethrow new InternalError(e);} } //transient Object[] elementData;這樣得到新的ArrayList對象,則是一個完全獨立的對象,包括對象屬性成員和原來對象沒有任何聯系。你走你的陽關道,我走我的獨木橋,你以后做什么事都影響不了我。這就是深拷貝。如果把上面的深拷貝改成淺拷貝,將會變成:
//淺拷貝public Object clone() {try {ArrayList<?> v = (ArrayList<?>) super.clone();//有公用的對象成員elementData,它就是連接兩個對象的罪魁禍首。v.elementData = elementData;v.modCount = 0;return v;} catch (CloneNotSupportedException e) {// this shouldn't happen, since we are Cloneablethrow new InternalError(e);} } //transient Object[] elementData;使用上面這個淺拷貝的clone()方法,原對象和克隆對象各自的對象成員elementData指向同一塊內存地址,藕斷絲連,相互影響著。你受傷了,我就難過;你過的開心,我就默默祝福你;你想我的時候,我也在想你。這就是淺拷貝。補充:如果A的對象成員屬性是自己定義的類型(記為B),A想要深拷貝,需要依靠B的深拷貝clone()方法。
題外話:
Cloneable 本身就是個比較雞肋的接口,盡量避免使用。如果一個類重寫了 Object 內定義的 clone() ,需要同時實現 Cloneable 接口(雖然這個接口內并沒有定義 clone() 方法),否則在調用 clone() 時會報 CloneNotSupportedException 異常,也就是說, Cloneable 接口只是個合法調用 clone() 的標識(marker-interface)。事實上,若想實現對象的克隆,你不得不重寫 Object 的 clone() 方法,還得實現 Cloneable 接口,這樣就有點麻煩了。簡單的做法,借助 Apache Commons 可以直接實現:
- 深克隆/拷貝(deep clone/copy): SerializationUtils
- 淺克隆/拷貝(shallow clone/copy):BeanUtils
簡單的克隆,也可以通過 copy-constructor (拷貝構造方法) 實現,詳情參考 Effective Java 作者 Josh Bloch 的這篇訪談:Copy Constructor Versus Cloning其中也談到了 Cloneable 接口的很多缺點。
總結
以上是生活随笔為你收集整理的重要接口—Cloneable接口的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 创建型模式—原型模式
- 下一篇: 重要接口—NavigableSet接口