Java深度復(fù)制功能與用法實例分析
本文實例講述了Java深度復(fù)制功能與用法。分享給大家供大家參考,具體如下:
寫在前面:
什么是深度復(fù)制?在Java里面,在創(chuàng)建一個對象,我們通常會有一個引用指向該對象,當(dāng)我們通過引用變量改變對象的值(屬性)時,引用是不變的,變的是內(nèi)存里面的那塊內(nèi)存,即引用所指向的對象。一般情況下,我們將該引用賦給另一個引用變量或者作為參數(shù)傳遞時,傳遞的也只是引用,即將引用指向“復(fù)制”了一份給另一個引用變量,隨后該引用變量也指向同一個對象,內(nèi)存里面并沒有創(chuàng)建一個新的對象。在某些情況下,我們需要“真正復(fù)制”對象,創(chuàng)建一份已知對象的copy,而不僅僅“復(fù)制”引用,用作備份也好,其他操作也好。
那么,該如何實現(xiàn)?
先說下思路:首先將對象序列化到流里,然后再反序列化,從流里讀取出來即可。
下面上代碼:
package com.yo.java;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;
import java.util.ArrayList;
import java.util.List;
/**
* Java 實現(xiàn)深度復(fù)制
* @author Yo
*
*/
public class DeepCopy implements Serializable{
int i;
/**
* @param args
* @throws IOException
* @throws ClassNotFoundException
*/
public static void main(String[] args) throws ClassNotFoundException, IOException {
demo1();
demo2();
}
/**
* 深度復(fù)制,實參類必須實現(xiàn)Serializable接口
* @param o
* @return
* @throws IOException
* @throws ClassNotFoundException
*/
public static Object deepCopy(Object o) throws IOException, ClassNotFoundException {
// //先序列化,寫入到流里
ByteArrayOutputStream bo = new ByteArrayOutputStream();
ObjectOutputStream oo = new ObjectOutputStream(bo);
oo.writeObject(o);
//然后反序列化,從流里讀取出來,即完成復(fù)制
ByteArrayInputStream bi = new ByteArrayInputStream(bo.toByteArray());
ObjectInputStream oi = new ObjectInputStream(bi);
return oi.readObject();
}
/**
* 引用傳遞與深度復(fù)制
* @throws ClassNotFoundException
* @throws IOException
*/
public static void demo1() throws ClassNotFoundException, IOException {
System.out.println("===========未使用深度復(fù)制=========");
DeepCopy dc1 = new DeepCopy();
dc1.i = 1;//初始化dc1里i的值
DeepCopy dc2 = dc1;
dc1.i = 2;//改變dc1里i的值
System.out.println("dc1 : " + dc1.i);
System.out.println("dc2(引用傳遞) : " + dc2.i);
System.out.println("===========使用深度復(fù)制=========");
DeepCopy dc3 = new DeepCopy();
dc3.i = 1;//初始化dc3里i的值
DeepCopy dc4 = (DeepCopy)deepCopy(dc3);
dc3.i = 2;//改變dc3里i的值
System.out.println("dc3 : " + dc3.i);
System.out.println("dc4(深度復(fù)制) : " + dc4.i);
}
/**
* 集合的值復(fù)制與深度復(fù)制
* @throws ClassNotFoundException
* @throws IOException
*/
public static void demo2() throws ClassNotFoundException, IOException {
System.out.println("===========未使用深度復(fù)制=========");
//創(chuàng)建DeepCopy對象,并初始化i的值,添加到list1集合
DeepCopy dc = new DeepCopy();
dc.i = 1;//初始化dc1里i的值
List<DeepCopy> list1 = new ArrayList<DeepCopy>();
list1.add(dc);
//未使用深度復(fù)制
List<DeepCopy> list2 = new ArrayList<DeepCopy>(list1);//這里與使用Collections.copy(dest, src)結(jié)果一樣
//改變list1中元素的值
for(DeepCopy d1 : list1) {
//改變dc1里i的值
d1.i = 2;
}
//遍歷list
for(DeepCopy d1 : list1) {
System.out.println("list1 : " + d1.i);
}
//遍歷list2
for(DeepCopy d2 : list2) {
System.out.println("list2(復(fù)制) : " + d2.i);
}
System.out.println("===========使用深度復(fù)制后=========");
DeepCopy dc3 = new DeepCopy();
dc3.i = 1;//初始化dc3里i的值
List<DeepCopy> list3 = new ArrayList<DeepCopy>();
list3.add(dc3);
List<DeepCopy> list4 = (List<DeepCopy>) deepCopy(list3);
for(DeepCopy d : list3) {
//改變dc3里i的值
d.i = 2;
}
for(DeepCopy d3 : list3) {
System.out.println("list3 : " + d3.i);
}
for(DeepCopy d4 : list4) {
System.out.println("list4(深度復(fù)制): " + d4.i);
}
}
}
以上運行結(jié)果如下(實測):
===========未使用深度復(fù)制========= dc1 : 2 dc2(引用傳遞) : 2 ===========使用深度復(fù)制========= dc3 : 2 dc4(深度復(fù)制) : 1 ===========未使用深度復(fù)制========= list1 : 2 list2(復(fù)制) : 2 ===========使用深度復(fù)制后========= list3 : 2 list4(深度復(fù)制): 1
由此可見,當(dāng)僅僅只是引用傳遞或者根據(jù)對象的值創(chuàng)建新的值,僅能稱為“淺復(fù)制”,當(dāng)原對象的屬性發(fā)生改變時,根據(jù)上面方式創(chuàng)建的新對象的屬性也會隨之改變;而如果采用深度復(fù)制,那是真正的copy了一份新的對象,新對象的與原對象不存在任何關(guān)聯(lián),原對象的屬性發(fā)生改變不會影響新對象,就如同copy的意義一樣
如上,如有不妥,如能指出,非常感謝
更多java相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《Java數(shù)據(jù)結(jié)構(gòu)與算法教程》、《Java操作DOM節(jié)點技巧總結(jié)》、《Java文件與目錄操作技巧匯總》和《Java緩存操作技巧匯總》
希望本文所述對大家java程序設(shè)計有所幫助。
相關(guān)文章
詳解springboot啟動時是如何加載配置文件application.yml文件
這篇文章主要介紹了詳解springboot啟動時是如何加載配置文件application.yml文件,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-06-06
springboot異步處理@NotBlank或@NotNull注釋校驗不生效問題
這篇文章主要介紹了springboot異步處理@NotBlank或@NotNull注釋校驗不生效問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-01-01
Java使用Validation自定義Double類型屬性校驗
這篇文章主要為大家詳細(xì)介紹了Java如何使用Validation自定義Double類型屬性校驗,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解下2024-11-11
SpringBoot條件注解之@ConditionalOnClass等注解的使用場景分析
文章詳細(xì)介紹了SpringBoot中條件注解的體系,包括基本概念、@ConditionalOnClass等常用注解的工作原理和使用場景,文章還探討了條件注解的組合使用、實戰(zhàn)應(yīng)用以及最佳實踐,幫助開發(fā)者更好地理解和應(yīng)用條件注解,實現(xiàn)更靈活和智能的應(yīng)用配置,感興趣的朋友一起看看吧2025-03-03
java并發(fā)之ArrayBlockingQueue詳細(xì)介紹
這篇文章主要介紹了java并發(fā)之ArrayBlockingQueue詳細(xì)介紹的相關(guān)資料,需要的朋友可以參考下2017-05-05
SpringBoot集成Druid配置(yaml版本配置文件)詳解
這篇文章主要介紹了SpringBoot集成Druid配置(yaml版本配置文件),本文通過實例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-12-12
SpringBoot配置GlobalExceptionHandler全局異常處理器案例
這篇文章主要介紹了SpringBoot配置GlobalExceptionHandler全局異常處理器案例,通過簡要的文章說明如何去進(jìn)行配置以及使用,需要的朋友可以參考下2021-06-06

