探討Java中的深淺拷貝問題
一、前言
拷貝這個(gè)詞想必大家都很熟悉,在工作中經(jīng)常需要拷貝一份文件作為副本。拷貝的好處也很明顯,相較于新建來說,可以節(jié)省很大的工作量。在Java中,同樣存在拷貝這個(gè)概念,拷貝的意義也是可以節(jié)省創(chuàng)建對(duì)象的開銷。
Object類中有一個(gè)方法clone(),具體方法如下:
protected native Object clone() throws CloneNotSupportedException;
1.該方法由 protected 修飾,java中所有類默認(rèn)是繼承Object類的,重載后的clone()方法為了保證其他類都可以正常調(diào)用,修飾符需要改成public。
2.該方法是一個(gè)native方法,被native修飾的方法實(shí)際上是由非Java代碼實(shí)現(xiàn)的,效率要高于普通的java方法。
3.該方法的返回值是Object對(duì)象,因此我們需要強(qiáng)轉(zhuǎn)成我們需要的類型。
4.該方法拋出了一個(gè)CloneNotSupportedException異常,意思就是不支持拷貝,需要我們實(shí)現(xiàn)Cloneable接口來標(biāo)記,這個(gè)類支持拷貝。
為了演示方便,我們新建兩個(gè)實(shí)體類Dept 和 User,其中User依賴了Dept,實(shí)體類代碼如下:
Dept類:
@Data
@AllArgsConstructor
@NoArgsConstructor
public class Dept {
private int deptNo;
private String name;
}
User類:
@Data
@AllArgsConstructor
@NoArgsConstructor
public class User {
private int age;
private String name;
private Dept dept;
}
二、淺拷貝
對(duì)于基本類型的的屬性,淺拷貝會(huì)將屬性值復(fù)制給新的對(duì)象,而對(duì)于引用類型的屬性,淺拷貝會(huì)將引用復(fù)制給新的對(duì)象。而像String,Integer這些引用類型,都不是不可變的,拷貝的時(shí)候會(huì)創(chuàng)建一份新的內(nèi)存空間來存放值,并且將新的引用指向新的內(nèi)存空間。不可變類型是特殊的引用類型,我們姑且認(rèn)為這些final類型的應(yīng)用也是復(fù)制值。

淺拷貝功能實(shí)現(xiàn)
@Data
@AllArgsConstructor
@NoArgsConstructor
public class User implements Cloneable{
private int age;
private String name;
private Dept dept;
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
如何驗(yàn)證我們的結(jié)論呢?首先對(duì)比被拷貝出的對(duì)象和原對(duì)象是否相等,不等則說明是新拷貝出的一個(gè)對(duì)象。其次修改拷貝出對(duì)象的基本類型屬性,如果原對(duì)象的此屬性發(fā)生了修改,則說明基本類型的屬性是同一個(gè),最后修改拷貝出對(duì)象的引用類型對(duì)象即Dept屬性,如果原對(duì)象的此屬性發(fā)生了改變,則說明引用類型的屬性是同一個(gè)。清楚測(cè)試原理后,我們寫一段測(cè)試代碼來驗(yàn)證我們的結(jié)論。
public static void main(String[] args) throws Exception{
Dept dept = new Dept(12, "市場(chǎng)部");
User user = new User(18, "Java旅途", dept);
User user1 = (User)user.clone();
System.out.println(user == user1);
System.out.println();
user1.setAge(20);
System.out.println(user);
System.out.println(user1);
System.out.println();
dept.setName("研發(fā)部");
System.out.println(user);
System.out.println(user1);
}
上面代碼的運(yùn)行結(jié)果如下
false
User{age=18, name='Java', dept=Dept{deptNo=12, name='市場(chǎng)部'}}
User{age=20, name='Java', dept=Dept{deptNo=12, name='市場(chǎng)部'}}
User{age=18, name='Java', dept=Dept{deptNo=12, name='研發(fā)部'}}
User{age=20, name='Java', dept=Dept{deptNo=12, name='研發(fā)部'}}
三、深拷貝
相較于淺拷貝而言,深拷貝除了會(huì)將基本類型的屬性復(fù)制外,還會(huì)將引用類型的屬性也會(huì)復(fù)制。

深拷貝功能實(shí)現(xiàn)
在拷貝user的時(shí)候,同時(shí)將user中的dept屬性進(jìn)行拷貝。
dept類:
@Data
@AllArgsConstructor
@NoArgsConstructor
public class Dept implements Cloneable {
private int deptNo;
private String name;
@Override
public Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
user類:
@Data
@AllArgsConstructor
@NoArgsConstructor
public class User implements Cloneable{
private int age;
private String name;
private Dept dept;
@Override
protected Object clone() throws CloneNotSupportedException {
User user = (User) super.clone();
user.dept =(Dept) dept.clone();
return user;
}
}
使用淺拷貝的測(cè)試代碼繼續(xù)測(cè)試,運(yùn)行結(jié)果如下:
false
User{age=18, name='Java旅途', dept=Dept{deptNo=12, name='市場(chǎng)部'}}
User{age=20, name='Java旅途', dept=Dept{deptNo=12, name='市場(chǎng)部'}}
User{age=18, name='Java旅途', dept=Dept{deptNo=12, name='研發(fā)部'}}
User{age=20, name='Java旅途', dept=Dept{deptNo=12, name='市場(chǎng)部'}}
除此之外,還可以利用反序列化實(shí)現(xiàn)深拷貝,先將對(duì)象序列化成字節(jié)流,然后再將字節(jié)流序列化成對(duì)象,這樣就會(huì)產(chǎn)生一個(gè)新的對(duì)象。
以上就是探討Java中的深淺拷貝問題的詳細(xì)內(nèi)容,更多關(guān)于Java深淺拷貝的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
mybatisplus下劃線駝峰轉(zhuǎn)換的問題解決
在mybatis-plus中,下劃線-駝峰自動(dòng)轉(zhuǎn)換可能導(dǎo)致帶下劃線的字段查詢結(jié)果為null,本文就來介紹一下mybatisplus下劃線駝峰轉(zhuǎn)換的問題解決,感興趣的可以了解一下2024-10-10
java如何給對(duì)象按照字符串屬性進(jìn)行排序
這篇文章主要介紹了java如何給對(duì)象按照字符串屬性進(jìn)行排序,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-11-11
詳解Java集合類之HashTable,Properties篇
這篇文章主要為大家詳細(xì)介紹一下Java集合類中HashTable和Properties的用法,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)Java有一定幫助,感興趣的可以了解一下2022-07-07
Spring aop 如何通過獲取代理對(duì)象實(shí)現(xiàn)事務(wù)切換
這篇文章主要介紹了Spring aop 如何通過獲取代理對(duì)象實(shí)現(xiàn)事務(wù)切換的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07
Java基于棧方式解決漢諾塔問題實(shí)例【遞歸與非遞歸算法】
這篇文章主要介紹了Java基于棧方式解決漢諾塔問題的方法,結(jié)合實(shí)例形式分析了java棧方式采用遞歸與非遞歸算法解決漢諾塔問題的相關(guān)操作技巧,需要的朋友可以參考下2017-11-11
詳解IDEA啟動(dòng)多個(gè)微服務(wù)的配置方法
這篇文章主要介紹了詳解IDEA啟動(dòng)多個(gè)微服務(wù)的配置方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-01-01
java 文件大數(shù)據(jù)Excel下載實(shí)例代碼
這篇文章主要介紹了java 文件大數(shù)據(jù)Excel下載實(shí)例代碼的相關(guān)資料,需要的朋友可以參考下2017-04-04
基于JavaSwing設(shè)計(jì)和實(shí)現(xiàn)的酒店管理系統(tǒng)
這篇文章主要介紹了基于JavaSwing+mysql的酒店管理系統(tǒng)設(shè)計(jì)和實(shí)現(xiàn),它可以實(shí)現(xiàn)酒店日常的管理功能包括開房、退房、房間信息、顧客信息管理等2021-08-08

