鱼C论坛

 找回密码
 立即注册
查看: 3922|回复: 4

[技术交流] Python的闭包和nonlocal关键字

[复制链接]
发表于 2019-8-31 01:13:42 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
本帖最后由 XiaoPaiShen 于 2019-8-31 12:45 编辑

最近从小甲鱼的视频中学习了Python的闭包,有些地方没有完全理解,就查了许多资料,终于弄懂了其中一些困扰我的问题,现在想写出来跟大家分享。
我们先来写一个简单的闭包如下:
  1. def closure(param):   

  2.     param = 20
  3.     def increase():        
  4.         param = 12
  5.         print("param value is {0}".format(param))        
  6.     return increase

  7. instance = closure(10)
  8. instance()  # param value is 12
  9. instance()  # param value is 12
复制代码


在上面的代码中,我们在嵌套函数中直接给param赋值来改变它是没有问题的,程序可以正常运行。因为无论你运行多少遍instance(),param的值都是不变的,程序输出都是一样的。
但是,如果我们把对param的修改变成每次运行使它的值递增1,就必须要使用nonlocal关键字了,否则程序就会抛出异常,我把程序修改如下:
  1. def closure(param):   

  2.     def increase():

  3.         nonlocal param
  4.         param = param + 1
  5.         print("param value is {0}".format(param))
  6.         
  7.     return increase

  8. instance = closure(10)
  9. instance()   # param value is 11
  10. instance()   # param value is 12

  11. another = closure(20)
  12. another()   # param value is 21
  13. another()   # param value is 22

  14. instance()  # param value is 13
  15. another()   # param value is 23
复制代码


在嵌套函数中,如果给变量param赋值,该变量就会成为increase函数中的本地变量,如果在它的外层有同名变量,就会被隐藏起来。
所以如果我们直接给嵌套函数中的本地变量赋值来改变它是没有问题的,但如果使用param = param + 1 来修改它,就属于在赋值前被引用,会抛出UnboundLocalError异常(本地变量在赋值前被引用)
这时我们就需要使用nonlocal关键字来声明变量param, 告诉编译器变量param在嵌套函数increase()中不是本地变量,就不会把外层的同名变量隐藏起来了。

另外,我们定义了两个实例,instance和another,分别赋予了不同的参数,一个是10,一个是20.
也就是我们给这两个实例定义了不同的运行环境,我们运行不同的实例,与他们相关的环境参数也都分别与各自的实例绑定,这就是‘闭包’名字的由来。

以上是我对闭包的理解,我也是刚刚接触,不知理解的是否正确,欢迎大家提出不同意见,相互交流。

点评

我很赞同!: 5.0
我很赞同!: 5
  发表于 2019-8-31 08:23
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

发表于 2019-8-31 08:38:33 | 显示全部楼层
谢谢楼主!!!
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2019-8-31 09:01:44 | 显示全部楼层
闭包就是保存局部信息的内容完整!
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2019-8-31 12:36:34 | 显示全部楼层
This is because when you make an assignment to a variable in a scope, that variable becomes local to that scope and shadows any similarly named variable in the outer scope.
在嵌套函数中,如果给变量var赋值,该变量就会成为inside函数中的本地变量,如果在它的外层有同名变量,就会被隐藏起来。
所以如果我们直接给嵌套函数中的本地变量赋值来改变它是没有问题的,但如果在赋值前引用它,或者使用 var += 1, 以及var = var + 1来修改它,都会抛出UnboundLocalError异常(本地变量在赋值前被引用)。
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-10-19 03:29

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表