每个单元格中带有字典的二维数组

时间:2017-11-13 20:55:16

标签: python list dictionary

我对以下代码有一点问题。我首先创建一个二维数组,我用字典填充(实际上每个单元格都是相同的字典)。现在,当我更新字典中值的单元格更改时,整个数组中的所有字典都会更改,而不仅仅是array[0][0]。请参阅以下代码:

dict_cell = {'item1': 20, 'item2': 25}
width = 2
height = 2
array = []
for i in range(height):
    row=[]
    for j in range(width):
        row.append(dict_cell)
    array.append(row)
array[0][0]['item1'] =2
print array

我得到以下输出:

[[{'item2': 25, 'item1': 2}, 
  {'item2': 25, 'item1': 2}], 
 [{'item2': 25, 'item1': 2}, 
  {'item2': 25, 'item1': 2}]]

当我想要:

[[{'item2': 25, 'item1': 2}, 
  {'item2': 25, 'item1': 20}], 
 [{'item2': 25, 'item1': 20}, 
  {'item2': 25, 'item1': 20}]]

有什么建议吗? 注意:我不能使用numpy。

2 个答案:

答案 0 :(得分:4)

python的工作方式是当你将dict_cell追加到row时,它会添加一个指向字典的指针。所以,实际上,你基本上连接了同一个字典!因此,当您修改其中一个条目时,所有字典都会发生变化,这并不足为奇。

解决此问题的最简单更改是使用dict_cell作为模板直接创建多个词典。用以下内容替换for循环中的append

    row.append(dict_cell.copy())

答案 1 :(得分:3)

  

让我们通过一个小例子来探讨你的问题:

所以假设你有一个清单:

a=['something']

还有第二个列表,其中包含:

list_1=[a,a,a]

所以当你这样做时:

a[0]="something_else"

您认为list_1的输出是什么?

我们来看看:

a=['something']
list_1=[a,a,a]


a[0]="something_else"
print(list_1)

输出:

[['something_else'], ['something_else'], ['something_else']]

因为在python变量中没有存储值,变量只是引用对象而对象存储值所以在list_1中所有变量都指向同一个对象:

检查:

for i in list_1:
    print(id(i))

输出:

4329477768
4329477768
4329477768

在你的情况下:

dict_cell = {'item1': 20, 'item2': 25}
width = 2
height = 2
array = []
for i in range(height):
    row=[]
    for j in range(width):
        row.append(dict_cell)
    array.append(row)
array[0][0]['item1'] =2



for item in array:
    if isinstance(item,list):
        for sub_item in item:
            print(id(sub_item))

输出:

4302653768
4302653768
4302653768
4302653768

因此您可以看到列表中的所有变量都指向同一个dict,因此如果您将任何内容更改为1,则会影响主要字典。

所以当你这样做时:

array[0][0]['item1'] =2

你不仅要修改数组的dict,你实际上修改了原始的dict,让我们检查一下:

dict_cell = {'item1': 20, 'item2': 25}
width = 2
height = 2
array = []
print("before modification {}".format(dict_cell))
for i in range(height):
    row=[]
    for j in range(width):
        row.append(dict_cell)
    array.append(row)
array[0][0]['item1'] =2

print("after  modification {}".format(dict_cell))

输出:

before modification {'item1': 20, 'item2': 25}
after  modification {'item1': 2, 'item2': 25}
  

好的,我遇到了问题,但解决方案是什么?

     

尝试深度扫描:

from copy import deepcopy
dict_cell = {'item1': 20, 'item2': 25}
width = 2
height = 2
array = []
for i in range(height):
    row=[]
    for j in range(width):
        row.append(deepcopy(dict_cell))
    array.append(row)
array[0][0]['item1'] =2
print(array)

输出:

[[{'item1': 2, 'item2': 25}, {'item1': 20, 'item2': 25}], [{'item1': 20, 'item2': 25}, {'item1': 20, 'item2': 25}]]
  

为什么deepcopy为什么不copy

假设你有这样的词:

dict_cell = {'item1': [20,34], 'item2': [25,9]}

您使用此dict_cell运行代码并获得输出:

[[{'item2': [25, 9], 'item1': 2}, {'item2': [25, 9], 'item1': [20, 34]}], [{'item2': [25, 9], 'item1': [20, 34]}, {'item2': [25, 9], 'item1': [20, 34]}]]

现在让我们尝试更改原始的dict值:

dict_cell = {'item1': [20,34], 'item2': [25,9]}
width = 2
height = 2
array = []
for i in range(height):
    row=[]
    for j in range(width):
        row.append(dict_cell.copy())
    array.append(row)
array[0][0]['item1'] =2


for key,value in dict_cell.items():
    value[0]='changed'

print(array,'\n')

输出:

[[{'item1': 2, 'item2': ['changed', 9]}, {'item1': ['changed', 34], 'item2': ['changed', 9]}], [{'item1': ['changed', 34], 'item2': ['changed', 9]}, {'item1': ['changed', 34], 'item2': ['changed', 9]}]] 

我们修改了orional dict但它改变了数组列表的dict中的内容,因为这是dict的浅层副本。它复制了dict但没有复制嵌套列表。

  

解决方案:

使用deepcopy。