>>> a, b = (1, 2, 3)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
ValueError: too many values to unpack (expected 2)
>>> a, b, c = (1, 2)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
ValueError: need more than 2 values to unpack
这意味着如果我们从上面调用foo函数并执行以下操作:
>>> x, y = foo()
…变量x等于字符串"hello",变量y等于字符串"world"。
所以最终,这意味着你最初的代码片段:
x = "salil"
y = "Ajay"
y, x = x, y
…在逻辑上等同于:
x = "salil"
y = "Ajay"
temp = (x, y) # evaluates to ("salil", "Ajay")
y, x = temp
……在逻辑上更是等同于:
x = "salil"
y = "Ajay"
temp = (x, y) # evaluates to ("salil", "Ajay")
y = temp[0]
x = temp[1]
这种机制的工作方式是两个特性的组合——形成隐式元组和元组/列表解包。
当您执行
something = x, y
操作时,Python将隐式地创建由x和y两个元素组成的tuple(一种不可变列表)。因此,以下两行代码完全等效:当然,元组可以包含两个以上的元素:
此功能的预期用例是为了使从函数返回多个值这样的操作更容易/更清晰:
第二个特性是tuple/list解包。每当左边有一系列变量,而另一边有任何类型的列表、元组或其他集合时,Python都会尝试将右边的每个元素与左边的元素匹配起来:
如果有帮助,那么
a, b, c = [11, 22, 33]
行与执行以下操作基本相同:注意,右边可以是任何类型的集合,而不仅仅是元组或列表。所以下面的代码是有效的:
(尽管如此,由于Python中的词典没有义务按照任何特定的顺序排列,而且由于密钥的顺序可以自由地进行置乱,因此解包它们可能不会有用,因为每次都可能得到不同的结果。)
如果集合中的变量数和元素数不匹配,Python将抛出异常:
这意味着如果我们从上面调用
foo
函数并执行以下操作:…变量
x
等于字符串"hello"
,变量y
等于字符串"world"
。所以最终,这意味着你最初的代码片段:
…在逻辑上等同于:
……在逻辑上更是等同于:
请注意,您可以认为这两个操作是分别进行的。首先形成元组并对其求值,然后将元组解压回变量中。最终的效果是两个变量的值是互换的。
(然而,事实证明,CPython解释器(Python的原始和“标准”实现)在这里做了一些优化:它将优化交换,而不会进行完整的元组解包——见下面的注释)。我不确定其他Python实现是否也会这样做,尽管我怀疑它们可能会这样做。在任何情况下,这种优化都是特定于实现的,并且独立于Python语言本身的设计。)
这适用于交换,因为首先计算
=
的右侧。所以在右边,它评估为
然后
x
和y
的赋值发生好吧,让我们看看:
这会产生:
请记住,Python是作为堆栈机运行的。在这种情况下,它优化了对
ROT_TWO
(即swap)指令的赋值。还有一个ROT_THREE
指令。但让我们试试别的:这产生了一般形式:
相关问题 更多 >
编程相关推荐