In Python, swapping the values of two variables using tuple assignment like (a, b) = (b, a) might seem like a simple trick, but how does it work internally without a temporary variable?
Python evaluates the right-hand side expression of the assignment separately from the left-hand side. Once the right-hand side is computed, its result is stored on the stack. Then, the left-hand side names are assigned using opcodes that operate directly on the stack.
For tuple assignments with two or three items, Python utilizes the stack directly using ROT_TWO or ROT_THREE opcodes to swap elements. For instance:
def foo(a, b): a, b = b, a
Disassembling this function using dis.dis(foo) reveals the following:
2 0 LOAD_FAST 1 (b) 3 LOAD_FAST 0 (a) 6 ROT_TWO 7 STORE_FAST 0 (a) 10 STORE_FAST 1 (b) 13 LOAD_CONST 0 (None) 16 RETURN_VALUE
Here, ROT_TWO swaps the top two values on the stack, effectively reversing the order of a and b.
For assignments involving more than three elements, Python constructs an intermediate tuple on the stack. Consider this example:
def bar(a, b, c, d): d, c, b, a = a, b, c, d
Its disassembly shows:
2 0 LOAD_FAST 0 (a) 3 LOAD_FAST 1 (b) 6 LOAD_FAST 2 (c) 9 LOAD_FAST 3 (d) 12 BUILD_TUPLE 4 15 UNPACK_SEQUENCE 4 18 STORE_FAST 3 (d) 21 STORE_FAST 2 (c) 24 STORE_FAST 1 (b) 27 STORE_FAST 0 (a) 30 LOAD_CONST 0 (None) 33 RETURN_VALUE
BUILD_TUPLE builds a tuple from the values on the stack in reverse order. Then, UNPACK_SEQUENCE pops the tuple and places its elements back onto the stack for assignment to the left-hand side variables.
While UNPACK_SEQUENCE appears redundant for two or three-item assignments, a later optimization step replaces the BUILD_TUPLE/UNPACK_SEQUENCE combo with the more efficient ROT_TWO or ROT_THREE opcodes, ensuring that the swapping mechanism is as streamlined as possible.
The above is the detailed content of How does Python swap tuple values without using a temporary variable?. For more information, please follow other related articles on the PHP Chinese website!