今天看Python习题,看到如下题目
def num(): return [lambda x: i*x for i in range(4)] print([m(2) for m in num()]) # 求输出结果是什么
我看了半天才明白这应该是一个列表生成式,列表中的元素为四个匿名函数,我本以为每个匿名函数应该是不一样的,因为他们的 i 不一样,所以应该返回的结果也会不同。可当我在命令行输出测试后才发现,完全不是这么回事啊!下面是输出结果
[6, 6, 6, 6] # 是真的6啊
后来去网上搜索了一下才知道,原来是作用域的问题,看来我Python作用域没有学好啊,这个匿名函数中的作用域和外层的循环作用域是不同的,匿名函数的引用了外层作用域变量 i ,当匿名函数被调用时,会输出 i * x,但是 i 在匿名函数的作用域中是不存在的,所有只能想外层作用域寻找,也就是for循环的作用域,但是此时循环早就已经结束了,i 也不再是当时生成该匿名的 i 了,现在的 i 已经是循环到最后一层固定不变的 i 了,也就是 i = 3, 所有匿名函数中的返回值就都是一样的了,即 3 * x。
如果还是不懂,可以将生成式转换为普通函数。如下
def num(): L = [] for i in range(4): def lambda_(x): return i * x # 该作用域内没有 i 变量,需要找寻上层作用域的 i 变量 L.append(lambda_) # 追加进列表的函数他们的返回值是 i * x,并不是0x,1x,2x,3x,变量还处于引用关系阶段 return L lambda_list = num() L = [] for lambda_ in lambda_list: result = lambda_(2) L.append(result) print(L)
python 的作用域都有legb规则,即Local,Enclose,Global,Builtin,作用域的查找顺序是从内向外的。