Python列表的深復制和淺復制示例詳解
一、深復制與淺復制
列表是Python中自帶的一種數據結構,在使用列表時,拷貝操作不可避免,下面簡單討論一下列表的深復制(拷貝)與淺復制
首先看代碼:
l1 = [5, 4, 3, 2, 1] # 用兩種方法實現對列表l1的拷貝 l2 = l1 l3 = l1[:] print(l1) # [5, 4, 3, 2, 1] print(l2) # [5, 4, 3, 2, 1] print(l3) # [5, 4, 3, 2, 1] #修改l1 l1[0] = 9 print(l1) # [9, 4, 3, 2, 1] print(l2) # [9, 4, 3, 2, 1] print(l3) # [5, 4, 3, 2, 1]
我們發現修改l1的第一個元素後,l2的第一個元素跟著改變,而l3並沒有發生變化。Python內置函數id()可以返回元素的地址,那麼我們使用這個函數來看一下三個列表的地址:
print(id(l1)) # 2927957162504 print(id(l2)) # 2927957162504 print(id(l3)) # 2927923243528
從結果來看,l1和l2地址是一樣的,也就是說l1和l2指向的是同一塊內存區域,顯然,通過 l2 = l1 操作,l1和l2都成瞭指向同一塊內存地址的“指針”,也就是說這個操作是為l1取瞭一個別名,也可以說l2是l1的一個引用。用一張圖來解釋:
那麼修改l1也就是在修改l2:
接下來看一下創建l3的過程,l3 = l1[:] ,這是將l1進行切片,並將切片後的列表拷貝到l3所指向的內存區域,同樣看圖:
也就是說l1和l3指向不同的內存區域,那麼修改l1並不會影響到l3:
通俗的來講,像l2這種,拷貝出來的對象和原對象的地址相同,為淺復制,像l3這種,分配新的內存空間並拷貝原始內容的,拷貝出來的對象和原對象的地址不同,為深復制。
二、復制列表內元素的淺復制
在復制列表中的所有元素的時候,進行淺復制
看一個比較有意思的東西,看代碼:
l1 = [1,2,3,[1,3]] l2 = l1[:] l1[3][1] = 9 print(l1) # [1, 2, 3, [1, 9]] print(l2) # [1, 2, 3, [1, 9]]
按照前面的理解,修改l1某個元素後,l2應該不會發生改變,可結果卻與我們預想的結果大相徑庭,於是,我們不得不思考一下l2深復制到底復制瞭什麼東西。實際上列表其實可以理解為一個“指針”,l1[3]是一個列表元素,l2[3]也是一個列表元素,執行以下代碼:
print(id(l2[3])) # 2014816956232 print(id(l1[3])) # 2014816956232
我們發現l1[3]和l2[3]指向的地址是一樣的,也就是說在執行 l2 = l1[:] 的時候,將一個地址拷貝瞭,所以修改l1[3]相當於修改l2[3],所以才會出現上述結果,這更加說明瞭列表其實就是一個指向一片內存區域的“指針”。那麼我們是不是可以說列表l2深復制l1,但是對列表中每個元素進行復制時進行的是淺復制呢?答案顯而易見。
修改l1[3]中的元素:
三、copy()和deepcopy()
copy模塊可以幫助我們實現對象的復制操作
列舉一下其他的拷貝列表的方式:
l4 = l1 * 1 print(id(l4)) # 2927957916296 l5 = list(l1) print(id(l5)) # 2927957767816 import copy l6 = copy.copy(l1) print(id(l6)) # 2927956854024 l7 = copy.deepcopy(l1) print(id(l7)) # 2927958503368
我們可以看到,這幾種拷貝方式所得到的的新對象與原對象的地址都不相同瞭,並沒有按照字面意思(copy進行淺復制,deepcopy進行深復制),那麼copy()和deepcopy()究竟有什麼區別呢,繼續看代碼:
list1 = [1,2,3,[1,3]] list2 = list1[:] list3 = copy.copy(list1) list4 = copy.deepcopy(list1) list1[3][0] = 9 print(list1) # [1, 2, 3, [9, 3]] print(list2) # [1, 2, 3, [9, 3]] print(list3) # [1, 2, 3, [9, 3]] print(list4) # [1, 2, 3, [1, 3]] print(id(list1[3])) # 2927923172616 print(id(list2[3])) # 2927923172616 print(id(list3[3])) # 2927923172616 print(id(list4[3])) # 2927967190728
可以發現copy()和前面提到的用 [:] 進行拷貝沒有本質上的區別,對列表中的每個元素進行復制時進行的是淺拷貝,而deepcopy()在復制列表中的每個元素的時候,進行的是深拷貝。
總結
到此這篇關於Python列表的深復制和淺復制的文章就介紹到這瞭,更多相關Python列表深復制和淺復制內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- 淺析Python的對象拷貝和內存佈局
- python淺拷貝與深拷貝使用方法詳解
- python中如何對多變量連續賦值
- python3.9之你應該知道的新特性詳解
- Python中list列表的賦值方法及遇到問題處理