美文网首页
理解Python中的闭包

理解Python中的闭包

作者: DamonYJ | 来源:发表于2017-11-06 20:03 被阅读43次

    Python基础

    Python中,函数也是一个对象,而且函数对象可以被赋值给变量,所以,通过变量也能调用该函数。
    例如:

    def now():
        print('2017-11-06')
    
    f = now
    f()    # 打印 2017-11-06
    

    理解什么是闭包?

    Python中,闭包的概念是:
    1.首先是个嵌套函数
    2.内层函数使用了外层函数的变量或者参数
    3.外层函数把内层函数当做返回值进行返回

    一段简单的闭包代码示例

    def func_outer(): # 外部函数
        
        temp = 666 # 外部函数的变量
    
        def func_inner(): # 嵌套的内部函数
            print(temp) # 使用了外部函数的变量
    
        return func_inner # 返回内部函数
        
    
    test = func_outer()
    test() # 打印666
    

    由以上可以看出,test变量其实就是返回的func_innter函数对象,test()就是通过这个变量来调用func_innter函数.

    闭包的注意点:

    请思考一下闭包中下面的几种情况

    1.在内层函数修改了外层函数的变量
    2.在闭包中,引用了一个可能会发生变化的变量

    先看第一种情况,对代码进行修改一下

    def func_outer():
        temp = 666
        def func_inner():
            temp = 888 # 0.在内层函数中修改了外层函数的变量的值
    
        print(temp)  # 1
        func_inner() #2. 在外部就执行了内部函数
        print(temp) # 3.
    
        return func_inner
    
    
    func_outer()
    

    在注释3处,打印的值还是666,这个问题比较简单,因为在注释0代码处相当于重新声明了一个局部变量temp,所以这只是一个作用域的问题(如果你是编程新手的话请打开你的搜索引擎了解作用域的知识)
    其实很简单,就四个大字: 就近原则

    如果要修改外部函数的temp变量使用关键字nonlocal声明:
    注释0处代码改为

    nonlocal temp
    temp = 888
    

    第二种情况,先看代码:

    def func_outer():
        temp = 666
        def func_inner():
            print(temp)
    
        # 修改了变量的值
        temp = 888    
        return func_inner
        
    test = func_outer()
    test()
    

    上面代码在声明内部函数func_inner()之后修改了temp变量的值,然而打印结果却是888.

    要想搞懂原因的话可以先写一个小测试

    def test_func():
        print(nondefine)
    

    上面函数打印了一个未被声明过的变量,但是,如果就这样运行的话程序时没有问题的,照样能正常运行.而当我们调用这个函数的时候test_func(),再次运行程序就报错了.报错原因

    NameError: name 'nondefine' is not defined
    

    上面的测试证明了当一个函数被调用时,它内部变量对应的值才会被确定,这就能解释上面为什么会打印888了.

    相关文章

      网友评论

          本文标题:理解Python中的闭包

          本文链接:https://www.haomeiwen.com/subject/fywlmxtx.html