第九天python3 闭包

Posted 潇湘神剑

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了第九天python3 闭包相关的知识,希望对你有一定的参考价值。

自由变量:未在本地作用域中定义的变量,例如定义在内存函数外的外层函数的作用域中的变量;

闭包:出现在嵌套函数中,指的是内层函数引用到了外层函数的自由变量,就形成了闭包;

示例1:

# -*- coding: utf-8 -*-
def counter():
    c = [0]
    print(id(c),"wai")
    def inc():
        c[0] += 1
        print(id(c),"nei")
        return c[0]
    return inc
foo = counter()
print(foo(),foo())
print(foo())

  代码解析:

  第四行不会报错,c已经在counter函数中定义过了,而且inc中的使用方式是为c的元素修改值,而不是重新定义;

  第八行打印1,2;

  第十行打印3,因为第九行的c和counter中的c不一样,而inc引用的是自由变量正式counter的函数;

  这是python2中实现闭包的方式,pyton3还可以使用nonlocal关键字;

示例2:

  

   上图代码使用global可以运行,但是这使用的是全局变量,而不是闭包;

   如果要对普通变量的闭包,python3中可以使用nonlocal;

nonlocal关键字

  使用了nonlocal关键字,将变量标记为在上级的局部作用域中定义,但不能在全局作用域中定义

示例:

      

  上图中count是外层函数的局部变量,被内部函数引用;内部函数使用nonlocal关键字声明count变量在上一级作用域中;

  左边代码可以正常使用,且形成闭包,右边代码不能正常运行,变量a不能在全局作用域中;

默认值作用域

  

  为什么第二次调用foo函数打印的是[1,1]?因为函数也是对象,python把函数的默认值放在了属性中,这个属性就是是伴随着这个函数对象的整个生命周期;如果print(xyz) #NameError,当前作用域没有xyz变量; 

  

  如上图所示,函数的地址并没有变,就是说函数这个对象没有变,调用它,它的属性__defaults__中使用元组保存所有默认值;xyz默认值是引用类型,引用类型的元素变动,并不是元组的变化;

 非引用类型例子

  

  如上图所示:属性__defaults__中使用元组保存所有默认值,它不会因为在函数体内使用了它而发生改变;

默认值的作用域

  可变类型默认值,如果使用默认值,就可能修改这个默认值;

  方法一:使用影子拷贝创建一个新的对象,永远不能改变传入的参数;

  

  如上图所示:函数体内,不改变默认值;xyz都是传入参数或者默认参数的副本,如果就想修改原参数,无能为力;

  方法二:通过值的判断就可以灵活的选择创建或者修改传入对象,这种方式灵活应用广泛,很多函数的定义,都可以看到使用None这个不可变的值作为默认参数,可以说这是一种惯用法;

   

  如上图所示:使用不可变类型默认值,如果使用缺省值None就创建一个列表,如果传入一个列表,就修改这个列表;

以上是关于第九天python3 闭包的主要内容,如果未能解决你的问题,请参考以下文章

python笔记第九天 装饰器

Python学习 第九天——选课系统

第九天

团队第二次冲刺第九天

python第九天

第一阶段冲刺(第九天)