stcaac 发表于 2017-5-5 09:45:54

python的闭包(3.0之前及之后)

  首先, 说说python里面的闭包吧:
  1. 需要函数嵌套, 就是一个函数里面再写一个函数.
  2. 外部函数需要返回一个内部函数的引用
  3. 外部函数中有一些局部变量, 并且, 这些局部变量在内部函数中有使用
  概念:
  1. 自由变量: 外部函数中定义的局部变量, 并且在内部函数中被使用.
  2. 闭包: 那个使用了自由变量并被返回的内部函数就称为闭包.
  一个例子:(来自<python核心编程>)
  def counter(start_at=0): count = def incr():count += 1return countreturn incr
  问题的提出: 关于python闭包的一个问题, 呵呵, 论坛一位网友提出的问题: 自由变量只能是list吗?
  def f():a = b = 1c = 'hello'd = (1, )e = Truef = {1: 2}def inf():print locals()a += 1b = 2c = 3d = (2, )e = Falsef = 3return areturn infa = f()t = a()print t
  上面这一段代码中, 内部函数inf第一句打印了locals(), 最终打印的结果只有a和f, 这段代码我的测试环境是python2.5, windows平台...对于这段, 我总结:
  1. 内部函数中, 遇到一个变量, 如果是试图改变它的子元素(集合类型)或自己名称空间(比如函数作为变量或对象作为变量时, 自己同时是名称空间)内部其他名称的引用时, 将会合法的引用到外部变量.
  2. 内部函数中, 遇到一个变量, 如果试图直接改变其自身, 多数情况都会得到"使用前未指定"的错误, 请看原因:
  def f():a = 1def inf():a += 1return areturn inf
  最重要的就在a+=1这一句, 这里, python实际上会认为你是在创建一个新的本地变量并为他赋值, 也就是说, python的解释器认为a+=1是a = a + 1, 而此时就很明显了, 本地变量中目前没有a, 所以, 就得到了我们上面说的那个错误...
  3. 通常需要在闭包内部去修改外部函数变量的时候, 我们需要借助一些集合类型或有名称空间的对象进行操作.
  以上这部分都是python3.0之前的情况, 在3.0之后, 加入了一个新的关键字nonlocal, 就像当初的global关键字解决函数内修改全局变量的问题一样漂亮, nonlocal将闭包内修改外部变量的问题也很优美的搞定了.
  def f():a1 = 'hello'a2 = 1a3 = o()a3.x = 1def inf():print(locals())nonlocal a2, a1a1 += ' world'a2 += 2a3.x += 1return a1, a2, a3return inf
  上面的代码中, 通过在闭包内, 使用外部变量之前, 用nonlocal去声明一下要在闭包中进行修改的那两个变量不是本地变量, 就OK了, 和global的用法完全一致, 这样, 我们就可以在其后的代码段中, 修改普通的外部变量了.....
页: [1]
查看完整版本: python的闭包(3.0之前及之后)