2022年 11月 9日

Python中序列的增量运算符+=的谜题

Python中序列的增量运算符+=的谜题

元组(tuple)是不可变的,列表(list)是可变的
那么元组里的某个元素是列表,那这个元素(列表)是可变的吗,显然也是可变的
如下操作,列表都是可以正常操作的,不会抛出异常

>>> t = (1,2, [3,4])
>>> t[2].append(5)
>>> t
(1, 2, [3, 4, 5])
>>> t[2].extend([6,7])
>>> t
(1, 2, [3, 4, 5, 6, 7])
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7

那么如果使用增量赋值运算符有+=呢,运行如下示例代码, 会发生什么?

A. t 变成 (1, 2, [30, 40, 50, 60])
B. 因为 tuple 不支持对它的元素赋值,所以会抛出 TypeError 异常
C. 以上两个都不是
D. A 和 B 都是对的
  • 1
  • 2
  • 3
  • 4
>>> t = (1,2,[3,4])
>>> t[2] += [5,6]
  • 1
  • 2

真实答案是 D,虽然抛异常了,但是也执行成功了。在控制台运行代码,显示结果如下:

>>> t = (1,2,[3,4])
>>> t[2] += [5,6]
Traceback (most recent call last):
  File "<input>", line 1, in <module>
TypeError: 'tuple' object does not support item assignment
>>> t
(1, 2, [3, 4, 5, 6])
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7

总结

  • 不要把可变对象放在元组里面。
  • 增量赋值不是一个原子操作,我们刚才也看到了,它虽然抛出了异常,但 t 的值还是改变了。