python copy.copy和copy.deepcopy

總的來講,就是copy.copy複製了對象,對於對象裏的元素,仍然保持引用。python

copy.deepcopy不只複製了對象,也複製了對象裏的元素,而不是引用。面試


1、前奏:熟悉Python內存管理app

在Python中,變量在第一次賦值時自動聲明,在建立---也就是賦值的時候,解釋器會根據語法和右側的操做數來決定新對象的類型。函數

引用計數器:一個內部跟蹤變量spa

引用計數:每個對象各有多少個引用.net

當對象被建立並(將其引用)賦值給變量時,該對象的引用計數就被設置爲 1code

>>> x = 3.14

語句 x=3.14,建立一個浮點型對象並將其引用賦值給了x,x是第一個引用,該對象的引用計數爲1對象

當一個對象(的引用)又被賦值到其餘變量,或作參數傳遞等,該對象的一個新的引用(或叫別名)被建立,則該對象的引用計數自動+1。blog

如下都會增長引用計數:內存

y = x   #作別名
foo(x)  #作參數傳遞
mylis = [1,2,x,'a'] #成爲容器對象的一個元素

 

如下都會減小引用計數:

複製代碼
del x   #del顯式銷燬

bar = x 
x = True    #對象的一個別名被賦值給其餘對象

mylis.remove(x) #對象被從窗口對象中移除

del mylis   #窗口對象自己被銷燬
複製代碼

 

2、Python的複製

從上面可見,對象的賦值其實是對象的引用。當建立一個對象,而後把它賦給另外一個變量的時候,python並無拷貝這個對象,而只是拷貝了這個對象的引用。

當你對一個對象賦值的時候(作爲參數傳遞,或者作爲返回值),Python和Java同樣,老是傳遞原始對象的引用,而不是一個副本。

複製代碼
"""傳遞原始對象的引用,而不是一個副本"""
a = [1,2,3]
b = a
b.append(100)
print b         #[1, 2, 3, 100]
print a         #[1, 2, 3, 100]
print id(a)     #11530368
print id(b)     #11530368 
複製代碼

如 果你想修改一個對象,並且想讓原始的對象不受影響,那你就須要對象複製。

能夠 使用copy.copy(),它能夠進行對象的淺複製(shallow copy),它複製了對象,但對於對象中的元素,依然使用引用.

(1)、使用切片[:]操做進行拷貝

(2)、使用工廠函數(如list/dir/set)等進行拷貝

(3)、copy.copy()

複製代碼
>>> jack = ['jack',['age',20]]
>>> tom = jack[:]
>>> anny = list(jack)
>>> jack
['jack', ['age', 20]]
>>> tom
['jack', ['age', 20]]
>>> anny
['jack', ['age', 20]]
>>> print id(jack),id(tom),id(anny)
13457088 18487376 18489136
複製代碼

接下來修改上面例子,對姓名和年級進行修改:

複製代碼
>>> tom[0]='tom'
>>> anny[0]='anny'
>>> print tom
['tom', ['age', 20]]
>>> print anny
['anny', ['age', 20]]
>>> anny[1][1]
20
>>> anny[1][1]= 18
>>> anny[1][1]
18
>>> print jack,tom,anny
['jack', ['age', 18]] ['tom', ['age', 18]] ['anny', ['age', 18]]
複製代碼

發現,雖然姓名都對號了,可是年齡卻都變成了18.這是爲何呢?

咱們看看它們元素的id

複製代碼
>>> [id(x) for x in jack]
[13463040, 13456608]
>>> [id(x) for x in tom]
[13463424, 13456608]
>>> [id(x) for x in anny]
[18501664, 13456608]
複製代碼

發現,其中列表中  姓名字符串  id都不同,可是 年齡列表id卻都相同。

這是由於:python中字符串不能夠修改,因此在爲tom和anny從新命名的時候,會從新建立一個’tom’和’anny’對象,替換舊的’jack’對象。

這就說明了,淺複製(shallow copy),它複製了對象,但對於對象中的元素,依然使用引用.

複製代碼
"""淺copy"""
import copy
aa = [1,2,3]
bb = copy.copy(aa)
print id(aa)    #11533088
print id(bb)    #12014776
bb[0] =100
print bb        #[100, 2, 3]
print aa        #[1,2,3]
#因爲數字不可變,修改的時候會替換舊的對象
print [id(x) for x in bb]   #[10247196, 10246388, 10246376]
print [id(y) for y in aa]   #[10246400, 10246388, 10246376]
複製代碼

下面試試對象中可變元素:

lis = [['a'],[1,2],['z',23]]
copyLis = copy.copy(lis)
copyLis[1].append('bar')
print copyLis   #[['a'], [1, 2, 'bar'], ['z', 23]]
print lis       #[['a'], [1, 2, 'bar'], ['z', 23]]

若是但願複製一個容器對象,以及它裏面的全部元素(包含元素的子元素),使用copy.deepcopy,這個方法會消耗一些時間和空間,不過,若是你須要徹底複製,這是惟一的方法.

"""深copy"""
deepLis = copy.deepcopy(lis)
deepLis[1].append('foo')    
print deepLis   #[['a'], [1, 2,'foo'], ['z', 23]]
print lis       #[['a'], [1, 2], ['z', 23]]


注意:

一、對於非容器類型(如數字、字符串、和其餘‘原子’類型的對象)沒有被拷貝一說。

二、若是元祖變量只包含原子類型對象,則不能深copy。

參考:http://blog.csdn.net/sharkw/article/details/1934090

http://www.01happy.com/python-shallow-copy-and-deep-copy/

相關文章
相關標籤/搜索