关于序列的+,*,+=,*=

2017-05-15  本文已影响1人  SHISHENGJIA

序列是支持 +* 操作的。通常 + 号两侧的序列由相同类型的数据所构成,在拼接的过程中,两个被操作的序列都不会被修改,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
上一篇 下一篇

猜你喜欢

热点阅读