脚本之家,脚本语言编程技术及教程分享平台!
分类导航

Python|VBS|Ruby|Lua|perl|VBA|Golang|PowerShell|Erlang|autoit|Dos|bat|

服务器之家 - 脚本之家 - Python - Python中闭包与lambda的作用域解析

Python中闭包与lambda的作用域解析

2022-07-14 14:03Python之简 Python

这篇文章主要介绍了Python中闭包与lambda的作用域解析,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

Python闭包与lambda的作用域

lambda写法

?
1
2
3
4
5
6
def fun():
    for i in range(3):
        yield lambda x : x * i
 
f0, f1, f2 = fun()
print(f0(1), f1(2), f2(3))

闭包的写法

?
1
2
3
4
5
6
7
8
9
def fun():
    result = []
    for i in range(3):
        def demo(x):
            return x * i
        result.append(demo)
    return result
f0, f1, f2 = fun()
print(f0(1), f1(2), f2(3))

上面两种写法的结果都是2,4,6,按最初的想法结果应该是0,2,6。

问题原因:

问题的本事在python的变量查找规则,LEGB(local,enclousing,global,bulitin),上面的例子中,i就是在闭包作用域(enclousing),而Python的闭包是迟绑定,闭包中用到的变量i的值,是在内部函数被调用时查找到的。

解决办法

将闭包作用域变为局部作用域

lambda写法

?
1
2
3
4
5
6
def fun():
    for i in range(3):
        yield lambda x, i = i: x * i
 
f0, f1, f2 = fun()
print(f0(1), f1(2), f2(3))

闭包写法

?
1
2
3
4
5
6
7
8
9
def fun():
    result = []
    for i in range(3):
        def demo(x, i=i):
            return x * i
        result.append(demo)
    return result
f0, f1, f2 = fun()
print(f0(1), f1(2), f2(3))

以上输出结果0,2,6

另一种情况:

?
1
2
3
4
5
def fun():
    for i in range(3):
        yield lambda x : x * i
f0, f1, f2 = fun()
print(f0(1), f1(2), f2(3))

输出结果还是2,4,6

问题原因

fun()方法返回的生成器(或迭代器),并没有真正的执行,而是在每次调用的时候进行执行。

在遍历后执行打印时,i这个变量使用的是最后调用的值。将lambda看作闭包方法的话,变量i值还是闭包作用域(no local)

python当中的坑(闭包与lambda)

先来看一个栗子

?
1
2
3
4
5
def create():
    return [lambda x:i*x for i in range(5)]
 
for i in create():
    print(i(2))

结果:

8

8

8

8

8

create函数的返回值时一个列表,列表的每一个元素都是一个函数 -- 将输入参数x乘以一个倍数i的函数。预期的结果时0,2,4,6,8. 但结果是5个8,意外不意外。

由于出现这个陷阱的时候经常使用了lambda,所以可能会认为是lambda的问题,但lambda表示不愿意背这个锅。问题的本质在与python中的属性查找规则,LEGB(local,enclousing,global,bulitin),在上面的例子中,i就是在闭包作用域(enclousing),而Python的闭包是 迟绑定 , 这意味着闭包中用到的变量的值,是在内部函数被调用时查询得到的

解决办法也很简单,那就是变闭包作用域为局部作用域。

?
1
2
3
4
5
def create():
    return [lambda x, i=i:i*x for i in range(5)]
 
for i in create():
    print(i(2))

换种写法:

?
1
2
3
4
5
6
7
8
9
10
def create():
    a = []
    for i in range(5):
        def demo(x, i=i):
            return x*i
        a.append(demo)
    return a
 
for i in create():
    print(i(2))

以上两种写法是一样的

结果:

0
2
4
6
8

再来一个问题相似的栗子

代码很简单:(声明:python3问题)

?
1
2
3
4
nums = range(2,20)
for i in nums:
    nums = filter(lambda x: x==i or x%i, nums)
print(list(nums))

结果:

[2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19]

同样按照正常的逻辑结果应该为:

[2, 3, 5, 7, 11, 13, 17, 19]

问题产生的原因:

  • 在python3当中filter()函数返回的是一个迭代器,因此并没有做真正的执行,而是在每次调用的时候执行(python2中filter()返回的值列表,无此现象)
  • 在遍历后执行打印时,现在执行循环当中的函数,同上面一个栗子的问题,i这个变量使用的是最后调用时的值,与以上栗子不同的是以上栗子用的是内嵌作用域的值,而这个栗子用的是全局i的值

修改代码:

?
1
2
3
4
nums = range(2,20)
for i in nums:
    nums = filter(lambda x,i=i: x==i or x%i, nums)
print(list(nums))

结果:

[2, 3, 5, 7, 11, 13, 17, 19]

以上为个人经验,希望能给大家一个参考,也希望大家多多支持服务器之家。

原文链接:https://blog.csdn.net/qq_1290259791/article/details/80895715

延伸 · 阅读

精彩推荐
  • PythonPython脚本调试工具安装过程

    Python脚本调试工具安装过程

    这篇文章主要介绍了Python脚本调试工具非常好用,本文给大家介绍pycharm工具的安装过程及使用详解,本文通过图文并茂的形式给大家介绍的非常详细,对大...

    宁宁的日常操作11932021-08-24
  • Python详解Python3中的Sequence type的使用

    详解Python3中的Sequence type的使用

    这篇文章主要介绍了详解Python3中的Sequence type的使用,是Python入门学习中的基础知识,需要的朋友可以参考下...

    wangdai5622020-07-28
  • Pythonpython开发之for循环操作实例详解

    python开发之for循环操作实例详解

    这篇文章主要介绍了python开发之for循环操作,以实例形式较为详细的分析了Python中for循环的具体使用技巧,具有一定参考借鉴价值,需要的朋友可以参考下 ...

    Hongten3402020-08-02
  • Python用Python编写一个每天都在系统下新建一个文件夹的脚本

    用Python编写一个每天都在系统下新建一个文件夹的脚本

    这篇文章主要介绍了用Python编写一个每天都在系统下新建一个文件夹的脚本,虽然这个实现听起来有点无聊...但却是学习os和time模块的一个小实践,需要的朋...

    Python教程网2122020-06-21
  • Pythonpython 正则表达式的使用

    python 正则表达式的使用

    这篇文章主要介绍了python 正则表达式的使用,Python 中正则表达式应用非常广泛,如数据挖掘、数据分析、网络爬虫、输入有效性验证等,Python 也提供了利用...

    Silent丿丶黑羽11542022-02-18
  • Python在Python中定义和使用抽象类的方法

    在Python中定义和使用抽象类的方法

    提起Java的抽象类大家都比较熟悉,Python中我们可以使用abc模块来构建抽象类,这里就为大家讲解在Python中定义和使用抽象类的方法...

    iamyangjy5642020-08-30
  • Pythonpython爬虫入门教程--HTML文本的解析库BeautifulSoup(四)

    python爬虫入门教程--HTML文本的解析库BeautifulSoup(四)

    Beautiful Soup是python的一个库,最主要的功能是从网页抓取数据。下面这篇文章主要给大家介绍了python爬虫之HTML文本的解析库BeautifulSoup的相关资料,文中介绍...

    FOOFISH2562020-11-12
  • PythonAnaconda多环境多版本python配置操作方法

    Anaconda多环境多版本python配置操作方法

    下面小编就为大家带来一篇Anaconda多环境多版本python配置操作方法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧...

    conda4692020-12-07