Python 链式赋值:为什么代码输出 (3, 2, 1) 而不是 (1, 2, 3)?
python 链式赋值机制解读
刚接触 python 的编程新手常常被链式赋值操作搞得一头雾水。例如:
a, b, c = b, c, a = c, b, a = 1, 2, 3print(a, b, c)
令人惊讶的是,代码输出结果并非 (1, 2, 3),而是 (3, 2, 1)。本文将深入剖析 python 中链式赋值的机制,解答这一令人困惑的问题。
python 中的赋值顺序
立即学习“Python免费学习笔记(深入)”;
不同于 c 语言,python 中的赋值是从右往左进行的。也就是说,在上述代码中,首先赋值给最右边的变量 a,然后再依次赋值给 b 和 c。
字节码解析
为了更直观地理解代码执行过程,我们可以使用 dis 模块查看字节码:
$ python -m dis ./main.py 1 0 LOAD_CONST 0 ((1, 2, 3)) 2 DUP_TOP 4 UNPACK_SEQUENCE 3 6 STORE_NAME 0 (a) 8 STORE_NAME 1 (b) 10 STORE_NAME 2 (c) 12 DUP_TOP 14 UNPACK_SEQUENCE 3 16 STORE_NAME 1 (b) 18 STORE_NAME 2 (c) 20 STORE_NAME 0 (a) 22 UNPACK_SEQUENCE 3 24 STORE_NAME 2 (c) 26 STORE_NAME 1 (b) 28 STORE_NAME 0 (a) 2 30 LOAD_NAME 3 (print) 32 LOAD_NAME 0 (a) 34 LOAD_NAME 1 (b) 36 LOAD_NAME 2 (c) 38 CALL_FUNCTION 3 40 POP_TOP 42 LOAD_CONST 1 (None) 44 RETURN_VALUE
字节码中,dup_top 指令用于复制栈顶元素。unpack_sequence 指令用于将序列(如元组)解包并将其元素压入栈中。
执行过程
- 压入元组 (1, 2, 3) 栈中:load_const 指令将元组 (1, 2, 3) 加载到栈中。
- 复制元组:dup_top 指令复制栈顶元素,即元组,在栈中形成两个相同的元组。
- 解包第一个元组并赋值:unpack_sequence 指令将栈顶元组解包并依次赋值给 a, b, c。栈中的第一个元组作为第一个值被弹出。
- 复制第二个元组:dup_top 指令再次复制栈顶元组,在栈中形成两个相同元组。
- 解包第二个元组并覆盖赋值:第二个 unpack_sequence 指令将栈顶元组解包并覆盖赋值给 b, c, a。栈中的第二个元组作为第二个值被弹出。
- 解包第三个元组并覆盖赋值:第三个 unpack_sequence 指令将栈顶元组解包并覆盖赋值给 c, b, a。栈中的第三个元组作为第三个值被弹出。
- 调用 print 函数并输出结果:最后,print 函数被调用,将变量 a, b, c 的值输出到控制台。
结论
简而言之,python 中链式赋值是 从右往左赋值 的,并且每个赋值操作都会覆盖前面的赋值结果。最终输出的结果反映了最后一个赋值操作的作用。