Python中對象的引用與復制


在python進行像b = a這樣的賦值時,只會創建一個對a的新引用,使a的引用計數加1,而不會創建新的對象:

>>> a = 'xyz'
>>> import sys
>>> sys.getrefcount(a)
3
>>> b = a
>>> sys.getrefcount(b)
4
>>> id(a)
88292288L
>>> id(b)
88292288L

這樣,當引用的對象是可變對象的時候(列表,字典,可變集合等),會產生意料之外的行為:

>>> a = [1, 2, 3, 4]
>>> b = a
>>> b.append(5)
>>> a
[1, 2, 3, 4, 5]

因為a和b引用的是同一對象,改變其中一個,另外一個也會隨之改變。當我們想建立一個副本而不是引用時,可以復制對象。

復制對象一般使用copy模塊:

>>> a = [1, 2, 3, 4]
>>> import copy
>>> b = copy.copy(a)
>>> b.append(5)
>>> b
[1, 2, 3, 4, 5]
>>> a
[1, 2, 3, 4]

這樣就可以了,但這種復制是一種淺復制,復制的新對象中包含的是對原始對象中的項的引用,如果對象的項為可變對象,也會產生不可控行為:

>>> a = [1, [1, 2]]
>>> b = copy.copy(a)
>>> b[1].append(3)
>>> b
[1, [1, 2, 3]]
>>> a
[1, [1, 2, 3]]

這時候就要使用深復制了。深復制將創建一個新對象,並遞歸地復制它所包含的所有對象:

>>> a = [1, [1, 2]]
>>> b = copy.deepcopy(a)
>>> b[1].append(3)
>>> b
[1, [1, 2, 3]]
>>> a
[1, [1, 2]]

對於不可改變的對象而言(字符串,數字,元組)等,沒有必要拷貝,因為它們是不可改變的,不用擔心會不經意間改動了它們。拷貝操作也只會得到原對象:

>>> a = (1, 2, 3)
>>> b = copy.copy(a)
>>> a is b
True

對於可變對象來(列表,字典,可變集合)來說,可以分別使用內置函數list(),dict(),set()來進行淺復制,速度是比使用copy模塊快的。

列表也可以使用切片進行淺復制:

>>> a = [1, 2, 3, 4]
>>> b = a[:]
>>> a is b
False
>>> b
[1, 2, 3, 4]

對序列數據類型(字符串,列表,元組)進行*操作時,也僅僅是復制了對象中項的引用,如果使用*創建一個多維列表:

>>> a = [1, 2, 3]
>>> b = [a]
>>> c = b * 3
>>> a.append(4)
>>> c
[[1, 2, 3, 4], [1, 2, 3, 4], [1, 2, 3, 4]]

最好是在列表推導中使用淺復制來創建多維列表,可以避免隱式的引用共享:

>>> a = [1, 2, 3]
>>> c = [list(a) for i in range(3)]
>>> a.append(4)
>>> c
[[1, 2, 3], [1, 2, 3], [1, 2, 3]]

  


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM