python中關(guān)于對象復(fù)制有三種類型的使用方式咆霜,賦值、淺拷貝與深拷貝嘶朱。他們既有區(qū)別又有聯(lián)系蛾坯,剛好最近碰到這一類的問題,研究下疏遏。
一脉课、賦值
在python中,對象的賦值就是簡單的對象引用财异,這點(diǎn)和C++不同倘零。如下:
list_a = [1,2,3,"hello",["python","C++"]]
list_b = list_a
這種情況下,list_b和list_a是一樣的宝当,他們指向同一片內(nèi)存视事,list_b不過是list_a的別名胆萧,是引用庆揩。
我們可以使用 list_b is list_a 來判斷,返回true跌穗,表明他們地址相同订晌,內(nèi)容相同。也可使用id(x) for x in list_a, list_b 來查看兩個(gè)list的地址蚌吸。
賦值操作(包括對象作為參數(shù)锈拨、返回值)不會(huì)開辟新的內(nèi)存空間,它只是復(fù)制了新對象的引用羹唠。也就是說奕枢,除了list_b這個(gè)名字以外,沒有其它的內(nèi)存開銷佩微。
修改了list_a缝彬,就影響了list_b;同理哺眯,修改了list_b就影響了list_a谷浅。
二、淺拷貝(shallow copy)
淺拷貝會(huì)創(chuàng)建新對象奶卓,其內(nèi)容是原對象的引用一疯。
淺拷貝有三種形式:切片操作,工廠函數(shù)夺姑,copy模塊中的copy函數(shù)
比如對上述list_a墩邀,
切片操作:list_b = list_a[:] 或者 list_b = [each for each in list_a]
工廠函數(shù):list_b = list(list_a)
copy函數(shù):list_b = copy.copy(list_a)
淺拷貝產(chǎn)生的list_b不再是list_a了,使用is可以發(fā)現(xiàn)他們不是同一個(gè)對象盏浙,使用id查看眉睹,發(fā)現(xiàn)它們也不指向同一片內(nèi)存留潦。但是當(dāng)我們使用 id(x) for x in list_a 和 id(x) for x in list_b 時(shí),可以看到二者包含的元素的地址是相同的辣往。
在這種情況下兔院,list_a和list_b是不同的對象,修改list_b理論上不會(huì)影響list_a站削。比如list_b.append([4,5])坊萝。
但是要注意,淺拷貝之所以稱為淺拷貝许起,是它僅僅只拷貝了一層十偶,在list_a中有一個(gè)嵌套的list,如果我們修改了它园细,情況就不一樣了惦积。
list_a[4].append("C")。查看list_b猛频,你將發(fā)現(xiàn)list_b也發(fā)生了變化狮崩。這是因?yàn)椋阈薷牧饲短椎膌ist鹿寻。修改外層元素睦柴,會(huì)修改它的引用,讓它們指向別的位置毡熏,修改嵌套列表中的元素坦敌,列表的地址并為發(fā)生變化,指向的都是同一個(gè)位置痢法。
三狱窘、深拷貝(deep copy)
深拷貝只有一種形式,copy模塊中的deepcopy函數(shù)财搁。
和淺拷貝對應(yīng)蘸炸,深拷貝拷貝了對象的所有元素,包括多層嵌套的元素妇拯。因而幻馁,它的時(shí)間和空間開銷要高。
同樣對list_a越锈,若使用list_b = copy.deepcopy(list_a)仗嗦,再修改list_b將不會(huì)影響到list_a了。即使嵌套的列表具有更深的層次甘凭,也不會(huì)產(chǎn)生任何影響稀拐,因?yàn)樯羁截惓鰜淼膶ο蟾揪褪且粋€(gè)全新的對象,不再與原來的對象有任何關(guān)聯(lián)丹弱。
四德撬、關(guān)于拷貝操作的警告
1铲咨、對于非容器類型,如數(shù)字蜓洪,字符纤勒,以及其它“原子”類型,沒有拷貝一說隆檀。產(chǎn)生的都是原對象的引用摇天。
2、如果元組變量值包含原子類型對象恐仑,即使采用了深拷貝泉坐,也只能得到淺拷貝。