Python 列表困惑

18 投票
3 回答
675 浏览
提问于 2025-04-16 17:24

假设我有以下代码:

a_list = [[0]*10]*10

这段代码会生成下面这个列表:

[[0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]]

接着我想修改第一个列表中的第一个元素:

a_list[0][0] = 23

我本来以为只会修改第一个列表的第一个元素,但实际上每个列表的第一个元素都被改了:

[[23, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [23, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [23, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [23, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [23, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [23, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [23, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [23, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [23, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [23, 0, 0, 0, 0, 0, 0, 0, 0, 0]]

我找到了一种其他的方法来表示我的数据,以避免这个问题,但这是为什么呢?为什么不是只有第一个列表被改变?当我做第二个 *10 时,Python 是不是实际上复制了第一个列表的地址,而不是分配一个新的内存块?

3 个回答

3

为什么只有第一个列表没有变化呢?

原因很简单,实际上只有一个列表,而不是10个 - 就像你已经猜到的那样:

In [1]: [[0]*10]*10
Out[1]:
[[0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
 [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]]

In [2]: map(id, _)
Out[2]:
[54094624,
 54094624,
 54094624,
 54094624,
 54094624,
 54094624,
 54094624,
 54094624,
 54094624,
 54094624]

如果你想创建10个列表,可以通过类似下面的表达式轻松实现:

[[0]*10 for x in xrange(10)]
4

列表里存的是对对象的引用。对列表进行乘法操作其实就是重复这些引用(也就是说,指向的是同一个对象!)。对于不可变对象(比如整数)来说,这样做没问题,但你得到的其实是对同一个列表的多个引用。

如果你想创建不同的列表,可以用这个方法 [[0]*10 for _ in xrange(10)]

15

你对复制地址的理解是对的。可以这样想:

sub_list = [0] * 10
a_list = [sub_list] * 10

这段代码实际上和你之前发的代码是一样的。这意味着,当你改变 a_list 中的任何一个元素时,其实是在改变同一个列表 sub_list。你可以通过输入以下内容来确认这一点:

a_list = [[0] * 10] * 10
for n in a_list:
    print id(n)

这样每个元素的结果都会是一样的。为了避免这个问题,你应该使用:

a_list = [[0] * 10 for _ in range(10)]

这样可以为 a_list 中的每个元素创建一个新的子列表。

撰写回答