在python中使用[[0] * n] * m遇到的坑

問題

想要像C++里面那樣創(chuàng)建一個(gè)m*n的矩陣,當(dāng)我使用[[0] * n] * m的時(shí)候遇到如下問題:

In[2]: l1 = [[0] * 3] * 4
In[3]: l2 = [[0] * 3 for _ in range(4)]
In[4]: l1[0][0] = 1
In[5]: l2[0][0] = 1
In[6]: l1
Out[6]: [[1, 0, 0], [1, 0, 0], [1, 0, 0], [1, 0, 0]]
In[7]: l2
Out[7]: [[1, 0, 0], [0, 0, 0], [0, 0, 0], [0, 0, 0]]

用id()查看一下內(nèi)存地址就知道L1中每一個(gè)元素是同一個(gè)內(nèi)存地址,而L2每一個(gè)元素的內(nèi)存地址是不一樣的:

In[8]: id(l1[0][0])
Out[8]: 140726462423440
In[9]: id(l1[1][0])
Out[9]: 140726462423440
In[10]: id(l2[0][0])
Out[10]: 140726462423440
In[11]: id(l2[1][0])
Out[11]: 140726462423408

解決方法

涉及到python中淺拷貝和深拷貝的概念。
在涉及到對(duì)矩陣中的值進(jìn)行重新賦值(大多用在動(dòng)態(tài)規(guī)劃的算法題里面)時(shí),最好時(shí)用以下兩種方式進(jìn)行拷貝:

# 采用不可變?cè)?l1 = [[1,2,3],[4,5,6]]
id(l1[0])
Out[13]: 2417349020360
id(l1[1])
Out[14]: 2417346673288

# 列表推導(dǎo)式
l2 = [[0] * 3 for _ in range(4)]
id(l2[0])
Out[16]: 2417346784456
id(l2[1])
Out[17]: 2417348988360

下面補(bǔ)充一下python的淺拷貝和深拷貝的知識(shí)

在Python中復(fù)制對(duì)象

在Python中,我們使用=運(yùn)算符來創(chuàng)建對(duì)象的副本。您可能會(huì)認(rèn)為這會(huì)創(chuàng)建一個(gè)新對(duì)象。沒有。它僅創(chuàng)建一個(gè)共享原始對(duì)象引用的新變量。

舉個(gè)例子,我們創(chuàng)建一個(gè)名為 old_list 并將對(duì)象引用傳遞給 new_list使用=運(yùn)算符。

示例1:使用=運(yùn)算符進(jìn)行復(fù)制

#Copy using = operator
old_list = [[1, 2, 3], [4, 5, 6], [7, 8, 'a']]
new_list = old_list

new_list[2][2] = 9

print('Old List:', old_list)
print('ID of Old List:', id(old_list))

print('New List:', new_list)
print('ID of New List:', id(new_list))

從輸出中可以看到兩個(gè)變量 old_list 和 new_list共享相同的ID,即140673303268168。

因此,如果需要修改 new_list 要么 old_list,更改在兩者中均可見。

本質(zhì)上,有時(shí)可能希望保持原始值不變,而僅修改新值,反之亦然。在Python中,有兩種創(chuàng)建副本的方法:

  1. 淺拷貝
  2. 深拷貝

為了使這些復(fù)制生效,我們使用了copy模塊。

復(fù)制模塊(copy module)

copyPython模塊用于淺層和深層復(fù)制操作。假設(shè)需要復(fù)制化合物列表說X。例如:

import copy
copy.copy(x)
copy.deepcopy(x)

在這里,copy()返回淺拷貝副本x。同樣,deepcopy()返回的深拷貝副本x。

淺拷貝

淺拷貝副本會(huì)創(chuàng)建一個(gè)新對(duì)象,該對(duì)象存儲(chǔ)原始元素的引用。

因此,淺拷貝副本不會(huì)創(chuàng)建嵌套對(duì)象的副本,而是僅復(fù)制嵌套對(duì)象的引用。這意味著復(fù)制過程本身不會(huì)遞歸或創(chuàng)建嵌套對(duì)象的副本。

示例2:使用淺拷貝副本創(chuàng)建副本

import copy

old_list = [[1, 2, 3], [4, 5, 6], [7, 8, 9]]
new_list = copy.copy(old_list)

print("Old list:", old_list)
print("New list:", new_list)

在上面的程序中,我們創(chuàng)建了一個(gè)嵌套列表,然后使用copy()方法對(duì)其進(jìn)行淺拷貝復(fù)制。

這意味著它將創(chuàng)建具有相同內(nèi)容的新的獨(dú)立對(duì)象。為了驗(yàn)證這一點(diǎn),我們同時(shí)打印兩個(gè)old_list 和 new_list。

確認(rèn) new_list 與...不同 old_list,我們嘗試將新的嵌套對(duì)象添加到原始對(duì)象并進(jìn)行檢查。

示例3:使用淺拷貝復(fù)制將[4,4,4]添加到old_list

import copy

old_list = [[1, 1, 1], [2, 2, 2], [3, 3, 3]]
new_list = copy.copy(old_list)

old_list.append([4, 4, 4])

print("Old list:", old_list)
print("New list:", new_list)

在上面的程序中,我們創(chuàng)建了的淺拷貝副本 old_list。的new_list 包含對(duì)存儲(chǔ)在其中的原始嵌套對(duì)象的引用 old_list。然后,我們添加新的列表即[4, 4, 4]到old_list。此新子列表未復(fù)制到new_list。

但是,當(dāng)您在其中更改任何嵌套對(duì)象時(shí) old_list,更改出現(xiàn)在 new_list。

示例4:使用淺拷貝添加新的嵌套對(duì)象

import copy

old_list = [[1, 1, 1], [2, 2, 2], [3, 3, 3]]
new_list = copy.copy(old_list)

old_list[1][1] = 'AA'

print("Old list:", old_list)
print("New list:", new_list)

在以上程序中,我們對(duì) old_list即old_list[1][1] = 'AA'。的兩個(gè)子列表old_list 和 new_list索引處[1][1]已修改。這是因?yàn)閮蓚€(gè)列表共享相同嵌套對(duì)象的引用。

深拷貝

深層副本將創(chuàng)建一個(gè)新對(duì)象,并以遞歸方式添加原始元素中存在的嵌套對(duì)象的副本。

讓我們繼續(xù)示例2。但是,我們將使用模塊中提供的deepcopy()功能來創(chuàng)建深層副本copy。深層副本創(chuàng)建原始對(duì)象及其所有嵌套對(duì)象的獨(dú)立副本。

示例5:使用deepcopy()復(fù)制列表

import copy

old_list = [[1, 1, 1], [2, 2, 2], [3, 3, 3]]
new_list = copy.deepcopy(old_list)

print("Old list:", old_list)
print("New list:", new_list)

在上面的程序中,我們使用deepcopy()函數(shù)來創(chuàng)建外觀相似的副本。

但是,如果您更改原始對(duì)象中的任何嵌套對(duì)象 old_list,您將看不到副本的任何更改 new_list。

示例6:使用深拷貝在列表中添加新的嵌套對(duì)象

import copy

old_list = [[1, 1, 1], [2, 2, 2], [3, 3, 3]]
new_list = copy.deepcopy(old_list)

old_list[1][0] = 'BB'

print("Old list:", old_list)
print("New list:", new_list)

在上面的程序中,當(dāng)我們將新值分配給 old_list,我們只能看到 old_list被修改。這意味著old_list和new_list是獨(dú)立的。這是因?yàn)閛ld_list 是遞歸復(fù)制的,這對(duì)于它的所有嵌套對(duì)象都是如此。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容