最近我开始好奇,但以下虚假 python 代码的第 2 行发生了什么:
def my_fun(foo,bar):
foo
return foo + bar
我开始感兴趣的原因是我正在尝试 Light Table 并试图在“foo”上放一个手表。它似乎导致 python 解释器挂起。
我认为这条线绝对没有效果并且不会引起任何错误,我是否正确?有人能解释一下口译员在这里到底做了什么吗?
最近我开始好奇,但以下虚假 python 代码的第 2 行发生了什么:
def my_fun(foo,bar):
foo
return foo + bar
我开始感兴趣的原因是我正在尝试 Light Table 并试图在“foo”上放一个手表。它似乎导致 python 解释器挂起。
我认为这条线绝对没有效果并且不会引起任何错误,我是否正确?有人能解释一下口译员在这里到底做了什么吗?
借助内置的dis模块,可以查看正在发生的事情:
import dis
def my_fun(foo,bar):
foo
return foo + bar
dis.dis(my_fun)
该dis.dis
函数反汇编函数(是的,它可以自行反汇编)、方法和类。
的输出dis.dis(my_fun)
是:
4 0 LOAD_FAST 0 (foo)
3 POP_TOP
5 4 LOAD_FAST 0 (foo)
7 LOAD_FAST 1 (bar)
10 BINARY_ADD
11 RETURN_VALUE
前两个字节码正是我们需要的:foo
行。
以下是这些字节码的作用:
基本上,foo
线没有效果。(好吧,如果foo
没有定义变量,那么LOAD_FAST
将抛出NameError
)
什么都没发生。它变得相当于一个毫无意义的操作查看dis
输出
In [3]: dis.dis(my_fun)
2 0 LOAD_FAST 0 (foo)
3 POP_TOP
3 4 LOAD_FAST 0 (foo)
7 LOAD_FAST 1 (bar)
10 BINARY_ADD
11 RETURN_VALUE
我们可以看到它可以快速加载,foo
然后什么也不做。
在命令行尝试一下:它只返回foo
. 这并不意味着它在某些特殊情况下不会产生副作用:如果您执行以下操作:
def my_fun(foo, bar):
foo.prop
return foo.func(bar)
即使从技术上讲我们刚刚返回了值,如果它被定义为一个属性,那么foo.prop
实际上可以调用一个函数。
但通常......你不会在模块中这样做,只能在交互式控制台中。
该foo
语句是表达式语句的示例,因此在解释器遇到它时会对其进行评估。
表达式语句计算表达式列表(可能是单个表达式)。
所以foo
加载,表达式被评估(它foo
本身,所以不需要进一步的操作),结果立即被遗忘。
什么都没发生:
>>> def baz(foo, bar):
foo
return bar
>>> baz(10, 20)
20
该声明无效。