我将尝试更深入地解释它。
如果你这样做
i = 0
f = lambda: i
您创建一个函数(lambda 本质上是一个函数),它访问其封闭范围的i
变量。
在内部,它通过一个包含i
. 粗略地说,它是一种指向实变量的指针,它可以在不同的时间点保存不同的值。
def a():
# first, yield a function to access i
yield lambda: i
# now, set i to different values successively
for i in range(100): yield
g = a() # create generator
f = next(g) # get the function
f() # -> error as i is not set yet
next(g)
f() # -> 0
next(g)
f() # -> 1
# and so on
f.func_closure # -> an object stemming from the local scope of a()
f.func_closure[0].cell_contents # -> the current value of this variable
在这里,所有的值i
- 在他们的时间 - 都存储在该闭包中。如果函数f()
需要它们。它从那里得到它们。
您可以在反汇编列表中看到这种差异:
这些说的功能a()
和f()
反汇编是这样的:
>>> dis.dis(a)
2 0 LOAD_CLOSURE 0 (i)
3 BUILD_TUPLE 1
6 LOAD_CONST 1 (<code object <lambda> at 0xb72ea650, file "<stdin>", line 2>)
9 MAKE_CLOSURE 0
12 YIELD_VALUE
13 POP_TOP
3 14 SETUP_LOOP 25 (to 42)
17 LOAD_GLOBAL 0 (range)
20 LOAD_CONST 2 (100)
23 CALL_FUNCTION 1
26 GET_ITER
>> 27 FOR_ITER 11 (to 41)
30 STORE_DEREF 0 (i)
33 LOAD_CONST 0 (None)
36 YIELD_VALUE
37 POP_TOP
38 JUMP_ABSOLUTE 27
>> 41 POP_BLOCK
>> 42 LOAD_CONST 0 (None)
45 RETURN_VALUE
>>> dis.dis(f)
2 0 LOAD_DEREF 0 (i)
3 RETURN_VALUE
b()
将其与看起来像的函数进行比较
>>> def b():
... for i in range(100): yield
>>> dis.dis(b)
2 0 SETUP_LOOP 25 (to 28)
3 LOAD_GLOBAL 0 (range)
6 LOAD_CONST 1 (100)
9 CALL_FUNCTION 1
12 GET_ITER
>> 13 FOR_ITER 11 (to 27)
16 STORE_FAST 0 (i)
19 LOAD_CONST 0 (None)
22 YIELD_VALUE
23 POP_TOP
24 JUMP_ABSOLUTE 13
>> 27 POP_BLOCK
>> 28 LOAD_CONST 0 (None)
31 RETURN_VALUE
循环的主要区别是
>> 13 FOR_ITER 11 (to 27)
16 STORE_FAST 0 (i)
在b()
与
>> 27 FOR_ITER 11 (to 41)
30 STORE_DEREF 0 (i)
in a()
:STORE_DEREF
存储在cell
对象(闭包)中,同时STORE_FAST
使用“正常”变量,它(可能)工作得更快一点。
lambda 也有所不同:
>>> dis.dis(lambda: i)
1 0 LOAD_GLOBAL 0 (i)
3 RETURN_VALUE
在这里你有一个LOAD_GLOBAL
,而上面的一个使用LOAD_DEREF
. 后者也用于关闭。
我完全忘记了lambda i=i: i
。
如果您将该值作为默认参数,它会通过完全不同的路径进入函数: 的当前值i
通过默认参数传递给刚刚创建的函数:
>>> i = 42
>>> f = lambda i=i: i
>>> dis.dis(f)
1 0 LOAD_FAST 0 (i)
3 RETURN_VALUE
这样该函数就被称为f()
. 它检测到缺少参数并用默认值填充相应的参数。这一切都发生在函数被调用之前;在函数中,您只会看到该值被获取并返回。
还有另一种方法可以完成您的任务:只需使用 lambda,就好像它需要一个值:lambda i: i
。如果你调用它,它会抱怨缺少参数。
但是您可以使用以下方法来应对functools.partial
:
ff = [functools.partial(lambda i: i, x) for x in range(100)]
ff[12]()
ff[54]()
这个包装器获取一个可调用的和许多要传递的参数。结果对象是一个可调用对象,它使用这些参数以及您提供给它的任何参数调用原始可调用对象。它可以在这里用来保持锁定到预期的值。