首頁  >  文章  >  後端開發  >  深入了解python中的copy模組(淺複製和深複製)

深入了解python中的copy模組(淺複製和深複製)

高洛峰
高洛峰原創
2017-03-28 17:16:321797瀏覽

主要是介紹python中的copy模組。

copy模組包括建立複合物件(包括列表、元組、字典和使用者定義物件的實例)的深淺複製的函數。

##copy(x)

建立新的複合物件並透過引用複製x的成員來建立x的淺複製。更加深層次說,

它複製了物件,但對於物件中的元素,依然使用引用。

對於內建類型,此函數並不經常使用。

而是使用諸如list(x), dict(x), set(x)等呼叫方式來建立x的淺複製,要知道像這樣

直接使用型別名稱顯然比使用copy()快很多。但是它們達到的效果是一樣的。

還有一點就是對於那些不可修改的物件(string, 數字, 元組),因為你不用擔心修改他們。複製不複製也就沒有

什麼大的意義了。

另外一點,你判斷物件之間是否為拷貝,可以使用is運算子來確定。

a is b -> True  a與b引用的是同一個對象,不是拷貝

       -> False a與b是彼此拷貝物件

例如下面

例如下面的例子,eg:

(1)

>>> a = [1,2,3]

>>> b = copy .copy(a)

>>> b

[1, 2, 3]

>>> a.append(4)

>>> a

[1, 2, 3, 4]

>>> b

[1, 2, 3]

>>> a is b

False

#(2)

>>> a = [1,2,3 ]

>>> b = a

>>> b

[1, 2, 3]

>> > a.append(4)

>>> a

#[1, 2, 3, 4]

>>> b

[1, 2, 3, 4]

>>> b.append(6)

>>> a, b

# ([1, 2, 3, 4, 6], [1, 2, 3, 4, 6])

(3)

>>> a = [1 ,2,3]

>>> b = list(a)

>>> b

[1, 2, 3]

>>> a.append(4)

>>> a

[1, 2, 3, 4]

> ;>> b

[1, 2, 3]

>>> 

(4)

>>> ; a = [[1], ['a'], ['A']]

>>> b = copy.copy(a)

>> > print a, b

[[1], ['a'], ['A']] [[1], ['a'], ['A']]

>>> b[1].append('b')

>>> b

[[1], ['a', 'b'] , ['A']]

>>> a

[[1], ['a', 'b'], ['A']]

>>> b.append([100,101])

#>>> b

[[1], ['a', 'b'], [ 'A'], [100, 101]]

>>> a######[[1], ['a', 'b'], ['A']] ######在(3)例子當中,我們可以看到a的淺複製對象b,它們是不同的對象,所以對對象的改變是不會######影響彼此的,但是這些a和b物件的元素是引用的同一個,所以a或b更改了它的物件的元素就會影響到###

另外一個的值。

如果你想要完全的拷貝一個物件和一個物件的所有元素的值,只有使用下面的deepcopy()函數。

deepcopy(x[, visit])

##透過建立新的複合物件並重複複製x的所有成員來建立x的深複製。

visit是一個可選的字典,目的是追蹤受存取的對象,從而檢測和避免重複定義

的資料結構中的循環。

儘管通常不需要,但是透過實作方法__copy__(self)和__deepcopy__(self, visit),

類別就可以實作自訂的複製方法,這兩個方法分別實現了淺複製和深複製操作。

__deepcopy__()方法必須使用字典visit,用來在複製過程中追蹤前面遇到的物件。對於

__deepcopy__()方法,除了將visit傳到實作中包含的其他deepcopy()方法(如果有的話)之外,

沒有必要在執行其他操作。

如果類別實作了pickle模組所使用的方法__getstate__()和__setstate__(),那麼copy模組將使用

這些方法來建立副本。

,但是透過實作方法__copy__(self)和__deepcopy__(self, visit),

類別就可以實作自訂的複製方法,這兩個方法分別實作了淺複製和深複製操作。

__deepcopy__()方法必須使用字典visit,用來在複製過程中追蹤前面遇到的物件。對於

__deepcopy__()方法,除了將visit傳到實作中包含的其他deepcopy()方法(如果有的話)之外,

沒有必要在執行其他操作。

如果類別實作了pickle模組所使用的方法__getstate__()和__setstate__(),那麼copy模組將使用

這些方法來建立副本。 ,但是透過實作方法__copy__(self)和__deepcopy__(self, visit),

類別就可以實現自訂的複製方法,這兩個方法分別實作了淺複製和深複製運算。

__deepcopy__()方法必須使用字典visit,用來在複製過程中追蹤前面遇到的物件。對於

__deepcopy__()方法,除了將visit傳到實作中包含的其他deepcopy()方法(如果有的話)之外,

沒有必要在執行其他操作。

如果類別實作了pickle模組所使用的方法__getstate__()和__setstate__(),那麼copy模組將使用

這些方法來建立副本。

eg:

>>> a = [[1], ['a'], ['A']]

#>>> ; import copy

>>> b = copy.deepcopy(a)

>>> b

[[1], ['a' ], ['A']]

>>> c = copy.copy(a)

>>> c

[[1] , ['a'], ['A']]

>>> a[1].append('b')

>>> a

[[1], ['a', 'b'], ['A']]

#>>> b######[[1], [' a'], ['A']]######>>> c#######[[1], ['a', 'b'], ['A']]# #####要注意的是:######(1)  copy模組用於像整數和字串這樣的簡單型別,不過很少需要這麼做。 ######(2)   這些複製函數無法與模組、類別物件、函數、方法、回溯、堆疊幀、檔案、套接字和其他類似類型同時運作。 ######如果無法複製對象,則會引發copy.error異常。 ###

以上是深入了解python中的copy模組(淺複製和深複製)的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn