python闭包与引用以及需要注意的陷阱


Posted in Python onSeptember 18, 2020

python闭包

关于闭包, 很多blog中都这样解释 :对于一个嵌套定义的函数,外层的函数的返回值是内层函数,而在内层函数中又引用了外层函数的局部变量,在外层函数执行后,其局部变量并非被回收,而会同返回的内层函数一同存在,而这一现象被称为闭包(closure)。

不过以上的理解有些繁琐和局限, 在计算机科学中 ,闭包(Closure)词法闭包(Lexical Closure)的简称,是引用了自由变量的函数。 这个被引用的自由变量将和这个函数一同存在,即使已经离开了创造它的环境也不例外。所以,有另一种说法认为闭包是由函数和与其相关的引用环境组合而成的实体。 也即对于第一段中的定义可以适当放开一些限制条件,python中的闭包实现也并非那么局限。

引用

通过上文介绍可以对于python闭包有大概的了解, 但是有些看似简单的细节却需要进一步阐述 。

python中变量的概念,这是与C/C++中极为不同的,在C/C++中变量是一个名称与内存合一的实体,改变一个变量的值,并不改变其内存的地址。 而变量这个概念在python中并不合用,很多场合它的运用都会让人混淆 。

python中所使用的概念是引用和对象,即如a=123,a即是一个引用名称,123是内存中所储存的对象值。这其实更像是C/C++中的指针与其所指向的内存,可以看作python在此之上对语法进行了包装。

回到之前讨论的闭包话题,在其中用到了 变量 的概念,即函数引用的 变量 将与函数一同存在,这里的 变量 其实是引用名称与内存对象的复合概念。我们这里对其进行进一步的阐明:

函数中所使用的外层函数引用名称(指针),在外层函数退出后其所指向的内存对象并不回收,而该引用名称(指针)会与内层函数一同存在,虽然此时该引用名称(指针)对于内层函数不是“可见的”。

陷阱

def count(): 
  fs = [] 
  for i in range(1, 4): 
    def f(): 
      return j*j 
    fs.append(f)
  return fs

f1, f2, f3 = count()
print(f1())
print(f2())
print(f3())

对于以上代码,假如按照C/C++中的概念去理解python中的变量,就会以为其输出依次为1、2、3。其实不然,真正输出为:3、3、3。根据上一小节中对于python中引用与闭包的阐述,在内存f函数中使用外层的引用名称i,在循环中虽然将不同的f函数加入到列表fs中,但是它们都使用的是同一个引用i,而该引用最后对应的值为3。

再看一段代码,这个会稍微复杂一点

def test():
  for i in range(4):
    yield i
    
g=test()

for n in [1,10]:
  g=(n+i for i in g)
  
print(list(g))

上面这段代码的输出,一时不查之下也会以为是11、12、13、14,而其真实结果却是20、21、22、23,让人一时抓不到头脑。首先在for循环中的生成器表达式(n+i for i in g),它其实本质上是一个函数,写成表达式的形式不过是一种语法糖,其函数形式为:

def gen(n):
  # g是外面全局的那个生成器g
  for i in g:
    yield n+i

即生成器generator本身是一种算法或是函数,只有在“调用”它的时候,也就是对其进行for或是list或是next之类的操作时,才会真正的有值流动。

那么对于以上第二例子中的代码,在for循环内n=1时,g这个生成器被重新赋值,但注意它此时只是一个特殊的函数,此时的n与i并没有真正相加,在for循环的第二轮n=10的时候,(n+i for i in g)表达式中对g才进行了调用,那么此时流进函数的n值其实是10,也就是此时g这个生成器对应的值为10、11、12、13,也就是i所引用的是这些值,下面又以相同的n+i的形式创造一个新的生成器对g重新赋值,并退出循环。则自然,此时g中对应的值为20、21、22、23.

以上就是python闭包与引用以及需要注意的陷阱的详细内容,更多关于python 闭包与引用的资料请关注三水点靠木其它相关文章!

Python 相关文章推荐
Linux中安装Python的交互式解释器IPython的教程
Jun 13 Python
python如何求解两数的最大公约数
Sep 27 Python
python 实现在tkinter中动态显示label图片的方法
Jun 13 Python
Django 数据库同步操作技巧详解
Jul 19 Python
python多线程与多进程及其区别详解
Aug 08 Python
Python generator生成器和yield表达式详解
Aug 08 Python
基于python3监控服务器状态进行邮件报警
Oct 19 Python
Python+numpy实现矩阵的行列扩展方式
Nov 29 Python
django配置app中的静态文件步骤
Mar 27 Python
Django中ORM的基本使用教程
Dec 22 Python
Pycharm 如何设置HTML文件自动补全代码或标签
May 21 Python
ubuntu安装jupyter并设置远程访问的实现
Mar 31 Python
python 代码运行时间获取方式详解
Sep 18 #Python
python判断字符串以什么结尾的实例方法
Sep 18 #Python
python绘制趋势图的示例
Sep 17 #Python
Python绘制组合图的示例
Sep 18 #Python
记一次Django响应超慢的解决过程
Sep 17 #Python
Visual Studio Code搭建django项目的方法步骤
Sep 17 #Python
Windows下pycharm安装第三方库失败(通用解决方案)
Sep 17 #Python
You might like
提升PHP执行速度全攻略(上)
2006/10/09 PHP
分享PHP入门的学习方法
2007/01/02 PHP
PHP常用开发函数解析之数组篇[未完结]
2012/07/30 PHP
关于php支持的协议与封装协议总结(推荐)
2017/11/17 PHP
php高清晰度无损图片压缩功能的实现代码
2018/12/09 PHP
ThinkPHP框架整合微信支付之JSAPI模式图文详解
2019/04/09 PHP
采用CSS和JS,刚好我最近有个站点要用到下拉菜单!
2006/06/26 Javascript
一个cssQuery对象 javascript脚本实现代码
2009/07/21 Javascript
基于Jquery的开发个代阴影的对话框效果代码
2011/07/28 Javascript
Javascript模块化编程详解
2014/12/01 Javascript
编写高性能Javascript代码的N条建议
2015/10/12 Javascript
Ionic如何实现下拉刷新与上拉加载功能
2016/06/03 Javascript
nodejs进阶(6)—连接MySQL数据库示例
2017/01/07 NodeJs
js实现带缓动动画的导航栏效果
2017/01/16 Javascript
php 修改密码实现代码
2017/05/24 Javascript
ionic3 懒加载
2017/08/16 Javascript
三分钟学会用ES7中的Async/Await进行异步编程
2018/06/14 Javascript
浅谈webpack性能榨汁机(打包速度优化)
2019/01/09 Javascript
vue实现滑动到底部加载更多效果
2020/10/27 Javascript
uni-app 支持多端第三方地图定位的方法
2020/01/03 Javascript
Python 3.6 读取并操作文件内容的实例
2018/04/23 Python
python调用java的jar包方法
2018/12/15 Python
Python生成器传参数及返回值原理解析
2020/07/22 Python
伦敦一家非常流行的时尚精品店:Oxygen Boutique
2017/01/15 全球购物
ROSEFIELD手表荷兰官方网上商店:北欧极简设计女士腕表品牌
2018/01/24 全球购物
在线购买廉价折扣书籍和小说:BookOutlet.com
2018/02/19 全球购物
乐高瑞士官方商店:LEGO CH
2020/08/16 全球购物
Linux常见面试题
2013/03/18 面试题
房屋改造计划书
2014/01/10 职场文书
2014五一国际劳动节活动总结范文
2014/04/14 职场文书
我的中国梦演讲稿500字
2014/08/19 职场文书
常务副县长“三严三实”对照检查材料思想汇报
2014/10/05 职场文书
一次性工伤赔偿协议书范本
2014/11/25 职场文书
农民工工资承诺书大全
2015/05/04 职场文书
防卫过当辩护词
2015/05/21 职场文书
在Django中使用MQTT的方法
2021/05/10 Python