假设我们有一个列表a=[1,2,3]
,我需要将a
的元素复制到新列表b
。
我们可以执行a=b
,但是a
和b
都指向同一列表。因此,对它们中的任何一个进行更改都会对两个列表进行更改。
>>> a=b
>>> a
[1, 2, 3]
>>> b
[1, 2, 3]
>>> b.append(4)
>>> a,b
([1, 2, 3, 4], [1, 2, 3, 4])
>>> a.append(5)
>>> a,b
([1, 2, 3, 4, 5], [1, 2, 3, 4, 5])
>>> a is b
True
>>> id(a),id(b)
(2287659980360, 2287659980360)
为避免这种情况,我们可以执行b=a[:]
。 a[:]
创建一个具有相同a
值的列表。现在,即使我突变了a
,b
也不会受到影响,反之亦然。 b
和b[:]
是两个不同的列表。
>>> b=a[:]
>>> a,b
([1, 2, 3, 4, 5], [1, 2, 3, 4, 5])
>>> a.append(6)
>>> a,b
([1, 2, 3, 4, 5, 6], [1, 2, 3, 4, 5])
>>> b.append(6)
>>> a,b
([1, 2, 3, 4, 5, 6], [1, 2, 3, 4, 5, 6])
>>> a is b
False
>>> id(a),id(b)
(2287659980360, 2287653308552)
[现在,如果我执行b[:]=a
,我的列表b
正在被突变。但是b[:]
是另一个列表吗? b
和b[:]
都指向不同的列表,对吗?如果我错了纠正我。如果我突变b
,为什么b[:]
被更改。我想念什么?
>>> a=['a','b','c']
>>> b[:]=a
>>> a,b
(['a', 'b', 'c'], ['a', 'b', 'c'])
>>> id(b),id(b[:])
(2287653308552, 2287660267080)
>>> b is b[:]
False
切片用作表达式或分配目标时,其含义有所不同。
作为表达式,求值b[:]
将产生一个新列表。但是,作为分配目标,分配给b[:]
并不意味着“求出表达式b[:]
然后分配给结果列表”。无论如何,您都无法分配给Python中的对象。
作为分配目标,分配给b[:]
会告诉现有列表将其内容替换为要分配给b[:]
的所有元素。可以通过__setitem__
方法来处理,例如b.__setitem__(slice(None, None), a)
。没有新的列表产生,并且b[:]
不作为表达式求值。
您可以在language reference documentation中的“如果目标是切片”下的分配语句中看到此说明。