尝试在一个地方更改时,值列表中的多个位置的值会发生变化

时间:2011-11-23 09:46:04

标签: python matrix

python中的以下代码:

matrix = [[0]*3]*2
matrix[0][1] = 1

看到要在所有matrix[][1]位置改变值,矩阵变为 [[0,1,0],[0,1,0]]代替[[0,1,0],[0,0,0]]

这是初始化变量的问题,还是python中的默认行为。

另外,我如何一次只更改一个值。

2 个答案:

答案 0 :(得分:6)

关于乘法运算符*如何在python中的列表上工作,有一些细微之处。

更一般地说,当应用于不可变对象(如整数或字符串)时,其效果会有所不同,以及在应用于可变对象(如列表和词典)时的效果如何

这可能会澄清问题:

>>> l = [0] * 3  #a list of immutable integers
>>> l
[0, 0, 0]
>>> l[0] = 1
>>> l
[1, 0, 0]
>>> l = [[0]] * 3  #a list of mutable lists
>>> l
[[0], [0], [0]]
>>> l[0][0] = 1
>>> l
[[1], [1], [1]]

编辑(评论中对@lazyr的称赞)在两种情况下,*运算符都会创建一个具有相同标识(内存地址)的对象列表,因此每个数组中的数字(和每个列表)实际上是同一个对象,但是,不可变类型不能被修改但只能被替换,因此当您尝试为整数数组分配新值时,您将事实上替换整个对象,而列表不是这种情况。上一个示例中的Bulding(请记住id函数返回对象的内存地址):

>>> m = [1] * 3
>>> id(m[0])
39356344
>>> id(m[1])
39356344
>>> m[1] = 2
>>> id(m[1])
39356320  # new memory addres = different object!
>>> m = [[1]] * 3
>>> id(m[0])
40275408
>>> id(m[1])
40275408
>>> m[1][0] = 2
>>> id(m[1])
40275408  # same memory address = still the same object!

因此,在您的情况下,可能的解决方法是初始化矩阵,如下所示:

>>> matrix = [[0 for i in range(3)] for j in range(2)]
>>> matrix
[[0, 0, 0], [0, 0, 0]]
>>> matrix[0][2] = 1
>>> matrix
[[0, 1, 0], [0, 0, 0]]

另一种 - 更激进的方式 - 将完全切换到numpy,这种方法更快,并且从根本上考虑了极快的矩阵和多维向量操作,但它也更难使用。

HTH!

答案 1 :(得分:3)

尝试:

matrix = [[0]*3 for j in xrange(2)]