Python:序列的增量賦值
阿新 • • 發佈:2018-10-05
nbsp 來看 http 原子 ror 謎題 不支持 實現 顯示
增量賦值運算符有 += 和 *=。+= 背後的特殊方法是 __iadd__,如果一個類沒有實現 __iadd__ 方法,Python 會退一步調用 __add__ 方法。這兩個方法的區別在於,__iadd__ 為就地改動,不會改變原值的內存地址,而 __add__ 方法會得到一個新對象。
考慮下面一個表達式:
a += b
如果 a 實現了 __iadd__ 方法,a 會就地改動(內存地址不變)。如果 a 沒有實現 __iadd__ 方法,那麽 a += b 這個表達式的效果就變得跟 a = a + b 一樣了,生成一個新的對象賦給 a。
總體來講,可變序列一般都實現了 __iadd__ 方法,因此 += 是就地加法,而不可變序列根本就不支持這個操作。
*= 和 += 一樣,只是背後的特殊方法為 __imul__。
a = [1, 2, 3]
b = [4, 5, 6]
print("id(a) = %d" % id(a))
a += b
print("id(a) = %d" % id(a))
c = [1, 2, 3]
print("id(c) = %d" % id(c))
c = c + b
print("id(c) = %d" % id(c))
d = (1, 2, 3)
print("id(d) = %d" % id(d))
d *= 2
print("id(d) = %d" % id(d))
運行結果如下:
id(a) = 1298277978824
id(a) = 1298277978824
id(c) = 1298277978696
id(c) = 1298277978632
id(d) = 1298277972872
id(d) = 1298277136616
了解了序列的增量賦值,我們來看 Leonardo Rochael 在 2013 年的 Python 巴西會議上提到的謎題:
t = (1, 2, [30, 40])
t[2] += [50, 60]
A. t 變成 (1, 2, [30, 40, 50, 60])
B. 因為 tuple 不支持對它的元素賦值,所以會拋出 TypeError 異常
C. 以上兩個都不是
D. A 和 B 都是對的
估計很多人會跟我一樣選 B,但其實答案是 D。在控制臺運行代碼,顯示結果如下:
總結:
1、不要把可變對象放在元組裏面。
2、增量賦值不是一個原子操作,我們剛才也看到了,它雖然拋出了異常,但 t 的值還是改變了。
Python:序列的增量賦值