第九天python3 闭包

chaigang 2020-06-27

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

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

示例1:

第九天python3 闭包

代码解析:

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

第八行打印1,2;

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

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

示例2:

第九天python3 闭包

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

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

nonlocal关键字

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

示例:

第九天python3 闭包    第九天python3 闭包

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

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

默认值作用域

第九天python3 闭包

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

第九天python3 闭包

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

 非引用类型例子

第九天python3 闭包

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

默认值的作用域

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

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

第九天python3 闭包

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

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

第九天python3 闭包 

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

相关推荐