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