我正在尝试在循环内创建函数:
functions = [] for i in range(3): def f(): return i # alternatively: f = lambda: i functions.append(f)
问题在于所有功能最终都相同。这三个函数都没有返回0、1和2,而是返回2:
print([f() for f in functions]) # expected output: [0, 1, 2] # actual output: [2, 2, 2]
为什么会发生这种情况,我应该怎么做才能获得分别输出0、1和2的3个不同函数?
你在后期绑定方面遇到了问题-每个函数都i尽可能晚地查找(因此,在循环结束后调用时,i将设置为2)。
可以通过强制早期绑定轻松修复:更改def f():为def f(i=i):以下形式:
def f():
def f(i=i):
def f(i=i): return i
缺省值(右手i输入i=i是参数名的默认值,i左手i输入i=i)是在def时间而不是在call时间查找的,因此从本质上讲,它们是一种专门查找早期绑定的方法。
i=i
如果你担心要f获得额外的参数(因此有可能被错误地调用),则有一种更复杂的方法,其中涉及将闭包用作“函数工厂”:
def make_f(i): def f(): return i return f
并在循环中使用f = make_f(i)而不是def语句。
f = make_f(i)