在Python中,經常要對一個list進行復制。對於復制,自然的就有深拷貝與淺拷貝問題。深拷貝與淺拷貝的區別在於,當從原本的list復制出的list之后,修改其中的任意一個是否會對另一個造成影響,即這兩個list在內存中是否儲存在同一個區域,這也是區分深拷貝與淺拷貝的重要依據。接下來我們就針對Python中list復制的幾種方法,來探究一下其是屬於深拷貝還是淺拷貝。弄清楚這個問題,有助於我們在編程中規避錯誤,減少不必要的調試時間。
一、非拷貝方法——直接賦值
如果用=直接賦值,是非拷貝方法。這兩個列表是等價的,修改其中任何一個列表都會影響到另一個列表。這也是Python作為動態語言與C這類靜態語言在思想上的不同之處。
old = [1,[1,2,3],3] new = old print('Before:') print(old) print(new) new[0] = 3 new[1][0] = 3 print('After:') print(old) print(new)
運行結果如下:

二、淺拷貝的幾種方法
1.copy()方法
我們來看以下代碼:
old = [1,[1,2,3],3] new = old.copy() print('Before:') print(old) print(new) new[0] = 3 new[1][0] =3 print('After:') print(old) print(new)
運行結果如下:
對於list的第一層,是實現了深拷貝,但對於嵌套的list,仍然是淺拷貝。這其實很好理解,內層的list保存的是地址,復制過去的時候是把地址復制過去了。嵌套的list在內存中指向的還是同一個。

2.使用列表生成式
使用列表生成式產生新列表也是一個淺拷貝方法,只對第一層實現深拷貝。
old = [1,[1,2,3],3] new = [i for i in old] print('Before') print(old) print(new) new[0] = 3 new[1][0] = 3 print('After') print(old) print(new)
運行結果如下:
3.用for循環遍歷
通過for循環遍歷,將元素一個個添加到新列表中。這也是一個淺拷貝方法,只對第一層實現深拷貝。
old = [1,[1,2,3],3] new = [] for i in range(len(old)): new.append(old[i]) print('Before:') print(old) print(new) new[0] = 3 new[1][0] = 3 print('After:') print(old) print(new)
運行結果如下:
4.使用切片
通過使用[:]切片,可以淺拷貝整個列表。同樣的,只對第一層實現深拷貝。
old = [1,[1,2,3],3] new = old[:] print('Before:') print(old) print(new) new[0] = 3 new[1][0] = 3 print('After:') print(old) print(new)
運行結果如下:
三、深拷貝的實現
如果用deepcopy()方法,則無論多少層,無論怎樣的形式,得到的新列表都是和原來無關的,這是最安全最清爽最有效的方法。
使用時,要導入copy。
import copy old = [1,[1,2,3],3] new = copy.deepcopy(old) print('Before:') print(old) print(new) new[0] = 3 new[1][0] = 3 print('After:') print(old) print(new)
運行結果如下:

