深入JAVA對(duì)象深度克隆的詳解
更新時(shí)間:2013年05月17日 09:52:26 作者:
本篇文章是對(duì)JAVA對(duì)象深度克隆進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
有時(shí)候,我們需要把對(duì)象A的所有值復(fù)制給對(duì)象B(B = A),但是這樣用等號(hào)給賦值你會(huì)發(fā)現(xiàn),當(dāng)B中的某個(gè)對(duì)象值改變時(shí),同時(shí)也會(huì)修改到A中相應(yīng)對(duì)象的值!
也許你會(huì)說,用clone()不就行了?!你的想法只對(duì)了一半,因?yàn)橛胏lone()時(shí),除了基礎(chǔ)數(shù)據(jù)和String類型的不受影響外,其他復(fù)雜類型(如集合、對(duì)象等)還是會(huì)受到影響的!除非你對(duì)每個(gè)對(duì)象里的復(fù)雜類型又進(jìn)行了clone(),但是如果一個(gè)對(duì)象的層次非常深,那么clone()起來非常復(fù)雜,還有可能出現(xiàn)遺漏!
既然用等號(hào)和clone()復(fù)制對(duì)象都會(huì)對(duì)原來對(duì)象產(chǎn)生影響,那么應(yīng)該怎么做才能實(shí)現(xiàn)復(fù)制后的對(duì)象不對(duì)原來對(duì)象有任何影響呢?
其實(shí)很簡(jiǎn)單,用對(duì)象的深度克隆,這種克隆實(shí)現(xiàn)了克隆后的對(duì)象和原來的對(duì)象是獨(dú)立開來的!
對(duì)象的深度克隆原理:將對(duì)象序列化后寫在輸出流里,因?yàn)閷懺诹骼锩娴膶?duì)象是一份拷貝,原對(duì)象仍然在JVM里;然后再把輸出流轉(zhuǎn)換為輸入流,把對(duì)象反序列化后寫出來!這樣就實(shí)現(xiàn)了對(duì)象的深度克隆,克隆后的兩個(gè)對(duì)象完全獨(dú)立開來,互不影響!
你會(huì)發(fā)現(xiàn)對(duì)象的深度克隆其實(shí)是利用的對(duì)象的序列化和反序列化,所以要進(jìn)行深度克隆的對(duì)象都要實(shí)現(xiàn)Serializable接口!
進(jìn)行深度克隆的實(shí)現(xiàn)代碼如下:
public Object copy() throws IOException, ClassNotFoundException{
ByteArrayOutputStream bos = new ByteArrayOutputStream();
ObjectOutputStream oos = new ObjectOutputStream(bos);
oos.writeObject(this);
ObjectInputStream ois = new ObjectInputStream(new ByteArrayInputStream(bos.toByteArray()));
return ois.readObject();
}
也許你會(huì)說,用clone()不就行了?!你的想法只對(duì)了一半,因?yàn)橛胏lone()時(shí),除了基礎(chǔ)數(shù)據(jù)和String類型的不受影響外,其他復(fù)雜類型(如集合、對(duì)象等)還是會(huì)受到影響的!除非你對(duì)每個(gè)對(duì)象里的復(fù)雜類型又進(jìn)行了clone(),但是如果一個(gè)對(duì)象的層次非常深,那么clone()起來非常復(fù)雜,還有可能出現(xiàn)遺漏!
既然用等號(hào)和clone()復(fù)制對(duì)象都會(huì)對(duì)原來對(duì)象產(chǎn)生影響,那么應(yīng)該怎么做才能實(shí)現(xiàn)復(fù)制后的對(duì)象不對(duì)原來對(duì)象有任何影響呢?
其實(shí)很簡(jiǎn)單,用對(duì)象的深度克隆,這種克隆實(shí)現(xiàn)了克隆后的對(duì)象和原來的對(duì)象是獨(dú)立開來的!
對(duì)象的深度克隆原理:將對(duì)象序列化后寫在輸出流里,因?yàn)閷懺诹骼锩娴膶?duì)象是一份拷貝,原對(duì)象仍然在JVM里;然后再把輸出流轉(zhuǎn)換為輸入流,把對(duì)象反序列化后寫出來!這樣就實(shí)現(xiàn)了對(duì)象的深度克隆,克隆后的兩個(gè)對(duì)象完全獨(dú)立開來,互不影響!
你會(huì)發(fā)現(xiàn)對(duì)象的深度克隆其實(shí)是利用的對(duì)象的序列化和反序列化,所以要進(jìn)行深度克隆的對(duì)象都要實(shí)現(xiàn)Serializable接口!
進(jìn)行深度克隆的實(shí)現(xiàn)代碼如下:
復(fù)制代碼 代碼如下:
public Object copy() throws IOException, ClassNotFoundException{
ByteArrayOutputStream bos = new ByteArrayOutputStream();
ObjectOutputStream oos = new ObjectOutputStream(bos);
oos.writeObject(this);
ObjectInputStream ois = new ObjectInputStream(new ByteArrayInputStream(bos.toByteArray()));
return ois.readObject();
}
相關(guān)文章
Java如何將字符串String轉(zhuǎn)換為整型Int
這篇文章主要介紹了Java如何將字符串String轉(zhuǎn)換為整型Int,文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的朋友可以參考一下2022-08-08解釋為什么Java中“1000==1000”為false而”100==100“為true
在日常編程中,我們經(jīng)常遇到一些看似簡(jiǎn)單卻隱藏著復(fù)雜邏輯的問題,這篇文章主要介紹了解釋為什么Java中“1000==1000”為false而”100==100“為true,需要的朋友可以參考下2024-01-01基于SpringBoot和MongoDB實(shí)現(xiàn)實(shí)時(shí)分析和日志處理功能
實(shí)時(shí)分析和日志處理在現(xiàn)代應(yīng)用程序開發(fā)中扮演著重要的角色,MongoDB是一個(gè)非常流行的NoSQL數(shù)據(jù)庫(kù),其高性能和靈活性使其成為實(shí)時(shí)分析和日志處理的理想選擇,本文將介紹如何使用?Spring?Boot?和?MongoDB?實(shí)現(xiàn)實(shí)時(shí)分析和日志處理的功能2023-06-06java?-jar/-cp啟動(dòng)添加外部的依賴包方式
這篇文章主要介紹了java?-jar/-cp啟動(dòng)添加外部的依賴包方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-01-01Java基礎(chǔ)總結(jié)之Thymeleaf詳解
Thymeleaf是一種現(xiàn)代的基于服務(wù)器端的Java模板引擎技術(shù),也是一個(gè)優(yōu)秀的面向Java的XML、XHTML、HTML5頁(yè)面模板,它具有豐富的標(biāo)簽語(yǔ)言、函數(shù)和表達(dá)式,在使用Spring Boot框架進(jìn)行頁(yè)面設(shè)計(jì)時(shí),一般會(huì)選擇Thymeleaf模板,需要的朋友可以參考下2021-05-05往DAO類中注入@PersistenceContext和@Resource的區(qū)別詳解
這篇文章主要介紹了往DAO類中注入@PersistenceContext和@Resource的區(qū)別詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-02-02