自由变量:未在本地作用域中定义的变量,例如定义在内存函数外的外层函数的作用域中的变量;
闭包:出现在嵌套函数中,指的是内层函数引用到了外层函数的自由变量,就形成了闭包;
示例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就创建一个列表,如果传入一个列表,就修改这个列表;