为什么多维数组应该用for循环构造?

时间:2012-03-05 06:43:13

标签: python

def make_set(rad):
    y = []
    for i in range(0, rad):
        x = []
        for j in range(0, rad):
            x.append(0)
        y.append(x)
    return y

def make_set_basic(rad):
    z = [[0] * rad] * rad
    return z

后一种实现(make_set_basic)有什么问题?

2 个答案:

答案 0 :(得分:6)

使用乘法运算符创建列表时,必须考虑所创建的对象是否可变或不可变。

在python中,整数是不可变的,在算术运算之后,会创建一个新对象:

>>> l = [0] * 3
>>> l
[0, 0, 0]
>>> map(id, l)
[42383312, 42383312, 42383312]
>>> l[0] += 1
>>> l
[1, 0, 0]
>>> map(id, l)
[42383288, 42383312, 42383312]

在上面的示例中,列表包含三个元素,所有元素都指向同一个对象(使用id内置函数进行检查)。但是,当修改第一个元素时,此元素现在指向另一个不同的对象,并且更改不会影响列表中的其他元素。

另一方面,如果对象是可变的,当您使用修改对象的方法时,您获得的结果就像修改应用于列表中的所有对象一样:

>>> l = [[0]] * 3
>>> l
[[0], [0], [0]]
>>> map(id, l)
[48544944, 48544944, 48544944]
>>> l[0].append(0)
>>> l
[[0, 0], [0, 0], [0, 0]]
>>> map(id, l)
[48544944, 48544944, 48544944]

请注意,在此示例中,在调用append方法之后,没有像上一个示例中那样创建新对象。鉴于列表中的所有元素仍然指向同一个对象(id builtin为所有元素返回相同的值),将为列表中的所有元素显示对象的新状态。

因此,回答您的问题,因为您可能不希望出现此行为,您需要在嵌套循环中创建所需的列表,以确保列表中的所有元素都指向您可以修改的其他对象不改变列表中其他元素的状态作为副作用。

答案 1 :(得分:3)

它创建对同一列表的rad个引用。改变其中一个元素会改变所有元素,因为它们是相同的列表。

[[0 for x in range(rad)] for y in range(rad)]