序列是支持 +
和 *
操作的。通常 +
号两侧的序列由相同类型的数据所构成,在拼接的过程中,两个被操作的序列都不会被修改,Python 会新建一个包含同样类型数据的序列来作为拼接的结果。
如果想要把一个序列复制几份然后再拼接起来,更快捷的做法是使用 *
。
>>> l = [1, 2, 3]
>>> l * 5
[1, 2, 3, 1, 2, 3, 1, 2, 3, 1, 2, 3, 1, 2, 3]
>>> 5 * 'abcd'
'abcdabcdabcdabcdabcd'
但是,在对序列进行乘法的时候需要注意,如果序列里的元素是其他可变对象的引用的话,会出现不是预期的结果。
>>> weird_board = [['_'] * 3] * 3
>>> weird_board
[['_', '_', '_'], ['_', '_', '_'], ['_', '_', '_']]
>>> weird_board[1][2] = 'O'
>>> weird_board
# 可以看出对其中一处的修改影响到了其他处,事实上里面的3个列表指向的是同一个引用
[['_', '_', 'O'], ['_', '_', 'O'], ['_', '_', 'O']]
上面的代码等同于下面的代码
row=['_'] * 3
board = []
for i in range(3):
board.append(row)
下面是正确的代码演示
board = [['_'] * 3 for i in range(3)]
上面的代码等同于下面的代码
board = []
for i in range(3):
row=['_'] * 3
board.append(row)
增量赋值运算符 +=
和 *=
的表现取决于它们的第一个操作对象。下面只讨论 +=
,*=
类似。
+=
背后的特殊方法是__iadd__
(就地加法),但如果类没有实现这个方法,那么会退一步调用__add__
。
另外对于不可变序列不支持上述的增量操作。
从下面的例子中可以看出,可变序列在运用增量乘法后ID不变,新元素追加到列表上。但是不可变序列元组在运用增量乘法后,创建了新的元组。(当然str
是个例外,CPython对其优化过,进行增量操作时并不会产生新的不可变序列)
l = [1, 2, 3]
print(id(l))
l += [4, 5, 6]
print(id(l))
l = (1, 2, 3)
print(id(l))
l += (4, 5, 6)
print(id(l))
47609336
47609336
9051392
9071280