探討Java中的深淺拷貝問題
一、前言
拷貝這個(gè)詞想必大家都很熟悉,在工作中經(jīng)常需要拷貝一份文件作為副本??截惖暮锰幰埠苊黠@,相較于新建來說,可以節(jié)省很大的工作量。在Java中,同樣存在拷貝這個(gè)概念,拷貝的意義也是可以節(jié)省創(chuàng)建對象的開銷。
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
對象,因此我們需要強(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; }
二、淺拷貝
對于基本類型的的屬性,淺拷貝會將屬性值復(fù)制給新的對象,而對于引用類型的屬性,淺拷貝會將引用復(fù)制給新的對象。而像String
,Integer
這些引用類型,都不是不可變的,拷貝的時(shí)候會創(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é)論呢?首先對比被拷貝出的對象和原對象是否相等,不等則說明是新拷貝出的一個(gè)對象。其次修改拷貝出對象的基本類型屬性,如果原對象的此屬性發(fā)生了修改,則說明基本類型的屬性是同一個(gè),最后修改拷貝出對象的引用類型對象即Dept
屬性,如果原對象的此屬性發(fā)生了改變,則說明引用類型的屬性是同一個(gè)。清楚測試原理后,我們寫一段測試代碼來驗(yàn)證我們的結(jié)論。
public static void main(String[] args) throws Exception{ Dept dept = new Dept(12, "市場部"); 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='市場部'}}
User{age=20, name='Java', dept=Dept{deptNo=12, name='市場部'}}
User{age=18, name='Java', dept=Dept{deptNo=12, name='研發(fā)部'}}
User{age=20, name='Java', dept=Dept{deptNo=12, name='研發(fā)部'}}
三、深拷貝
相較于淺拷貝而言,深拷貝除了會將基本類型的屬性復(fù)制外,還會將引用類型的屬性也會復(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; } }
使用淺拷貝的測試代碼繼續(xù)測試,運(yùn)行結(jié)果如下:
false
User{age=18, name='Java旅途', dept=Dept{deptNo=12, name='市場部'}}
User{age=20, name='Java旅途', dept=Dept{deptNo=12, name='市場部'}}
User{age=18, name='Java旅途', dept=Dept{deptNo=12, name='研發(fā)部'}}
User{age=20, name='Java旅途', dept=Dept{deptNo=12, name='市場部'}}
除此之外,還可以利用反序列化實(shí)現(xiàn)深拷貝,先將對象序列化成字節(jié)流,然后再將字節(jié)流序列化成對象,這樣就會產(chǎn)生一個(gè)新的對象。
以上就是探討Java中的深淺拷貝問題的詳細(xì)內(nèi)容,更多關(guān)于Java深淺拷貝的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Java壓縮和解壓縮ZIP文件實(shí)戰(zhàn)案例
這篇文章主要給大家介紹了關(guān)于Java壓縮和解壓縮ZIP文件的相關(guān)資料,ZIP是一種較為常見的壓縮形式,最近項(xiàng)目中遇到了再Java中壓縮和解壓縮zip文件的需求,特此分享給大家,需要的朋友可以參考下2023-07-07java中Spring Security的實(shí)例詳解
這篇文章主要介紹了java中Spring Security的實(shí)例詳解的相關(guān)資料,spring security是一個(gè)多方面的安全認(rèn)證框架,提供了基于JavaEE規(guī)范的完整的安全認(rèn)證解決方案,需要的朋友可以參考下2017-09-09基于SpringBoot實(shí)現(xiàn)自定義插件的流程詳解
在SpringBoot中,插件是一種擴(kuò)展機(jī)制,它可以幫助我們在應(yīng)用程序中快速地添加一些額外的功能,在本文中,我們將介紹如何使用 SpringBoot實(shí)現(xiàn)自定義插件,需要的朋友可以參考下2023-06-06解決java 分割字符串成數(shù)組時(shí),小圓點(diǎn)不能直接進(jìn)行分割的問題
這篇文章主要介紹了解決java 分割字符串成數(shù)組時(shí),小圓點(diǎn)不能直接進(jìn)行分割的問題,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-12-12基于Java實(shí)現(xiàn)中文分詞系統(tǒng)的示例代碼
這篇文章主要為大家詳細(xì)介紹了如何利用Java語言實(shí)現(xiàn)一個(gè)簡易的中文分詞系統(tǒng),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以嘗試一下2022-07-07