python 序列增量赋值谜一般的题目

说到python的增量赋值,大家里面就想到 +=, *= 之类的

+=背后的特殊方法是 iadd 意思是:就地加法,如果一个类没有实现这个方法,那么python会退一步使用 add来进行相加

a += b

如果a实现了就地相加方法,就是调用这个方法,同时对可变序列来说,就是直接改动,就像调用a.extend(b)一样,但是如果a没有这个方法,那就执行 a = a + b, 总的来说,可变序列一般都实现了就低价法,而不变序列根部就不支持这个操作,也不可能实现这个方法。

现在我们得出的结论是:
可变序列可以增量赋值,不可变序列不可以增量赋值
这个结论到底对不对?我们来看一个题:

t = (1,2,[3,4])
t[2] += [5,6]

用我们刚才的结论来判断,因为t是个tuple,是不变序列,所以不支持增量赋值,所以这个肯定会报错,没错,没错,我们用刚才的结论成功的做出了这题,来看报错信息:

<ipython-input-43-c823147bfbc0> in <module>()
----> 1 t[2] += [5,6]

TypeError: 'tuple' object does not support item assignment

哈哈,很简单,接着,我们再来看一下此时t的值:

In [44]: t
Out[44]: (1, 2, [3, 4, 5, 6])

你肯定会打呼:what the f**k!!!!,什么鬼,怎么还是变了,擦
结果: t被改动了,同时也抛出了错误

来吧,我们来看看这其中的原理
首先我们看看python字节码对于s[a] += b这种类型的解析

In [46]: import dis

In [47]: dis.dis('s[a] += b')
  1           0 LOAD_NAME                0 (s)
              3 LOAD_NAME                1 (a)
              6 DUP_TOP_TWO  
              7 BINARY_SUBSCR  ------1
              8 LOAD_NAME                2 (b)
             11 INPLACE_ADD    ---------2
             12 ROT_THREE
             13 STORE_SUBSCR  ------------3
             14 LOAD_CONST               0 (None)
             17 RETURN_VALUE

三个步骤(对应代码里面的1,2,3):
1.将s[a]存入栈顶
2.完成s[a] += b
3.存贮结果

有结果可以看出前两个步骤成功执行了,在最后一步报错,因为s是不可变的序列所以s[a] 赋值失败报错,通过这个例子我们可以得到的结论如下:
1.尽量不要把可变对象放到元组里面(通过extend方法可以避免这个问题)
2.增量赋值 操作不是原子操作,因为报错了,但是还是完成了 += 操作
这个问题是个不常遇见的边界问题,不常见,但是了解一下还是很有用处的~

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • http://python.jobbole.com/85231/ 关于专业技能写完项目接着写写一名3年工作经验的J...
    燕京博士阅读 7,632评论 1 118
  • 个人笔记,方便自己查阅使用 Py.LangSpec.Contents Refs Built-in Closure ...
    freenik阅读 67,777评论 0 5
  • Python简介 Python历史 Python 是由 Guido van Rossum 在八十年代末和九十年代初...
    莫名其妙的一生阅读 1,073评论 0 2
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,780评论 18 399
  • Python 是一种相当高级的语言,通过 Python 解释器把符合语法的程序代码转换成 CPU 能够执行的机器码...
    Python程序媛阅读 1,956评论 0 3