自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

細(xì)說 Java 的深拷貝和淺拷貝

開發(fā) 開發(fā)工具
任何變成語言中,其實都有淺拷貝和深拷貝的概念,Java 中也不例外。在對一個現(xiàn)有的對象進(jìn)行拷貝操作的時候,是有淺拷貝和深拷貝之分的,他們在實際使用中,區(qū)別很大,如果對其進(jìn)行混淆,可能會引發(fā)一些難以排查的問題。

 [[200191]]

一、前言

任何變成語言中,其實都有淺拷貝和深拷貝的概念,Java 中也不例外。在對一個現(xiàn)有的對象進(jìn)行拷貝操作的時候,是有淺拷貝和深拷貝之分的,他們在實際使用中,區(qū)別很大,如果對其進(jìn)行混淆,可能會引發(fā)一些難以排查的問題。

本文就在 Java 中的深拷貝和淺拷貝做一個詳細(xì)的解說。

二、什么是淺拷貝和深拷貝

首先需要明白,淺拷貝和深拷貝都是針對一個已有對象的操作。那先來看看淺拷貝和深拷貝的概念。

在 Java 中,除了基本數(shù)據(jù)類型(元類型)之外,還存在 類的實例對象 這個引用數(shù)據(jù)類型。而一般使用 『 = 』號做賦值操作的時候。對于基本數(shù)據(jù)類型,實際上是拷貝的它的值,但是對于對象而言,其實賦值的只是這個對象的引用,將原對象的引用傳遞過去,他們實際上還是指向的同一個對象。

而淺拷貝和深拷貝就是在這個基礎(chǔ)之上做的區(qū)分,如果在拷貝這個對象的時候,只對基本數(shù)據(jù)類型進(jìn)行了拷貝,而對引用數(shù)據(jù)類型只是進(jìn)行了引用的傳遞,而沒有真實的創(chuàng)建一個新的對象,則認(rèn)為是淺拷貝。反之,在對引用數(shù)據(jù)類型進(jìn)行拷貝的時候,創(chuàng)建了一個新的對象,并且復(fù)制其內(nèi)的成員變量,則認(rèn)為是深拷貝。

所以到現(xiàn)在,就應(yīng)該了解了,所謂的淺拷貝和深拷貝,只是在拷貝對象的時候,對 類的實例對象 這種引用數(shù)據(jù)類型的不同操作而已。

總結(jié)來說:

1、淺拷貝:對基本數(shù)據(jù)類型進(jìn)行值傳遞,對引用數(shù)據(jù)類型進(jìn)行引用傳遞般的拷貝,此為淺拷貝。

2、深拷貝:對基本數(shù)據(jù)類型進(jìn)行值傳遞,對引用數(shù)據(jù)類型,創(chuàng)建一個新的對象,并復(fù)制其內(nèi)容,此為深拷貝。

三、Java 中的 clone()

3.1 Object 上的 clone() 方法

在 Java 中,所有的 Class 都繼承自 Object ,而在 Object 上,存在一個 clone() 方法,它被聲明為了 protected ,所以我們可以在其子類中,使用它。

而無論是淺拷貝還是深拷貝,都需要實現(xiàn) clone() 方法,來完成操作。

 

可以看到,它的實現(xiàn)非常的簡單,它限制所有調(diào)用 clone() 方法的對象,都必須實現(xiàn) Cloneable 接口,否者將拋出 CloneNotSupportedException 這個異常。最終會調(diào)用 internalClone() 方法來完成具體的操作。而 internalClone() 方法,實則是一個native 的方法。對此我們就沒必要深究了,只需要知道它可以 clone() 一個對象得到一個新的對象實例即可。

而反觀 Cloneable 接口,可以看到它其實什么方法都不需要實現(xiàn)。對他可以簡單的理解只是一個標(biāo)記,是開發(fā)者允許這個對象被拷貝。

3.2 淺拷貝

先來看看淺拷貝的例子。

首先創(chuàng)建一個 class 為 FatherClass ,對其實現(xiàn) Cloneable 接口,并且重寫 clone()方法。

然后先正常 new 一個 FatherClass 對象,再使用 clone() 方法創(chuàng)建一個新的對象。

 

***看看輸出的 Log :

  1. I/cxmyDev: fatherA == fatherB : false  
  2. I/cxmyDev: fatherA hash : 560973324  
  3. I/cxmyDev: fatherB hash : 560938740  
  4. I/cxmyDev: fatherA name : 張三  
  5. I/cxmyDev: fatherB name : 張三 

可以看到,使用 clone() 方法,從 == 和 hashCode 的不同可以看出,clone() 方法實則是真的創(chuàng)建了一個新的對象。

但這只是一次淺拷貝的操作。

來驗證這一點,繼續(xù)看下去,在 FatherClass 中,還有一個 ChildClass 的對象 child ,clone() 方法是否也可以正常復(fù)制它呢?改寫一個上面的 Demo。

看到,這里將其內(nèi)的 child 進(jìn)行負(fù)責(zé),用起來看看輸出的 Log 效果。

  1. I/cxmyDev: fatherA == fatherB : false 
  2. I/cxmyDev: fatherA hash : 560975188 
  3. I/cxmyDev: fatherB hash : 560872384 
  4. I/cxmyDev: fatherA name : 張三 
  5. I/cxmyDev: fatherB name : 張三 
  6. I/cxmyDev: ================== 
  7. I/cxmyDev: A.child == B.child : true 
  8. I/cxmyDev: fatherA.child hash : 560891436 
  9. I/cxmyDev: fatherB.child hash : 560891436 

從***對 child 的輸出可以看到,A 和 B 的 child 對象,實際上還是指向了同一個對象,只對對它的引用進(jìn)行了傳遞。

3.3 深拷貝

既然已經(jīng)了解了對 clone() 方法,只能對當(dāng)前對象進(jìn)行淺拷貝,引用類型依然是在傳遞引用。

那么,如何進(jìn)行一個深拷貝呢?

比較常用的方案有兩種:

  1. 序列化(serialization)這個對象,再反序列化回來,就可以得到這個新的對象,無非就是序列化的規(guī)則需要我們自己來寫。
  2. 繼續(xù)利用 clone() 方法,既然 clone() 方法,是我們來重寫的,實際上我們可以對其內(nèi)的引用類型的變量,再進(jìn)行一次 clone()。

繼續(xù)改寫上面的 Demo ,讓 ChildClass 也實現(xiàn) Cloneable 接口。

最重要的代碼就在 FatherClass.clone() 中,它對其內(nèi)的 child ,再進(jìn)行了一次 clone() 操作。

再來看看輸出的 Log。

  1. I/cxmyDev: fatherA == fatherB : false 
  2. I/cxmyDev: fatherA hash : 561056732 
  3. I/cxmyDev: fatherB hash : 561057344 
  4. I/cxmyDev: fatherA name : 張三 
  5. I/cxmyDev: fatherB name : 張三 
  6. I/cxmyDev: ================== 
  7. I/cxmyDev: A.child == B.child : false 
  8. I/cxmyDev: fatherA.child hash : 561057304 
  9. I/cxmyDev: fatherB.child hash : 561057360 

可以看到,對 child 也進(jìn)行了一次拷貝,這實則是對 ChildClass 進(jìn)行的淺拷貝,但是對于 FatherClass 而言,則是一次深拷貝。

其實深拷貝的思路都差不多,序列化也好,使用 clone() 也好,實際上都是需要我們自己來編寫拷貝的規(guī)則,最終實現(xiàn)深拷貝的目的。

如果想要實現(xiàn)深拷貝,推薦使用 clone() 方法,這樣只需要每個類自己維護(hù)自己即可,而無需關(guān)心內(nèi)部其他的對象中,其他的參數(shù)是否也需要 clone() 。

四、總結(jié)

到現(xiàn)在基本上就已經(jīng)梳理清楚,Java 中淺拷貝和深拷貝的概念了。

實則淺拷貝和深拷貝只是相對的,如果一個對象內(nèi)部只有基本數(shù)據(jù)類型,那用 clone() 方法獲取到的就是這個對象的深拷貝,而如果其內(nèi)部還有引用數(shù)據(jù)類型,那用 clone() 方法就是一次淺拷貝的操作。

【本文為51CTO專欄作者“張旸”的原創(chuàng)稿件,轉(zhuǎn)載請通過微信公眾號聯(lián)系作者獲取授權(quán)】

戳這里,看該作者更多好文

責(zé)任編輯:武曉燕 來源: 51CTO專欄
相關(guān)推薦

2021-07-16 12:33:24

Javascript深拷貝淺拷貝

2009-05-19 17:28:44

深拷貝淺拷貝clone()

2021-09-27 11:07:11

深拷貝淺拷貝內(nèi)存

2022-07-26 08:07:03

Python淺拷貝深拷貝

2018-09-26 14:37:17

JavaScript前端編程語言

2024-03-15 15:03:23

2020-08-03 08:24:26

原型模式拷貝

2021-01-08 06:15:09

深拷貝淺拷貝寫時拷貝

2020-10-12 08:35:22

JavaScript

2023-05-17 08:42:46

深拷貝Golang

2018-05-10 14:20:18

前端JavaScript深拷貝

2022-09-30 15:03:09

C語言深拷貝淺拷貝

2024-02-05 22:56:16

C++拷貝開發(fā)

2022-11-07 11:37:27

深拷貝淺拷貝底層

2024-04-17 09:01:08

Python深拷貝淺拷貝

2023-05-17 07:36:00

淺拷貝深拷貝對象

2020-06-23 08:41:47

JavaScript開發(fā)技術(shù)

2023-09-22 12:21:33

Python深拷貝淺拷貝

2021-09-10 07:41:06

Python拷貝Python基礎(chǔ)

2019-02-25 08:58:16

Python深拷貝淺拷貝
點贊
收藏

51CTO技術(shù)棧公眾號