python嵌套函数、闭包与decorator
1
一段代码的执行结果不光取决与代码中的符号,更多地是取决于代码中符号的意义,而运行时的意义是由名字空间决定的。名字空间是在运行时由python虚拟机动态维护的,但是有时候我们希望能将名字空间静态化。即:我们希望有的代码不受名字空间变换的影响,始终保持一致的行为和结果。
这样的意义何在呢?
这就不得不说说嵌套函数了。

上面代码中,我们只设置了一次基准值。此后,在每次进行比较操作的时候,尽管调用的实际函数real_compare的local名字空间中没有base,而global名字空间中有base = 1,但是函数调用结果显示,real_compare以一种神奇的方式得知了base应该是10,而不是1。
也就是说在real_compare函数作为返回值被传递给compare_with_10的时候,有一个名字空间已经与real_compare紧紧地绑定在一起了,在执行real_compare的时候,这个名字空间又被恢复了,这就是一种将名字空间静态化的方法。
这个名字空间和函数捆绑后的结果被称为一个闭包(closure)。
比如PyFunctionObject是Python虚拟机专门为字节码指令准备的大包袱,global名字空间、默认参数都能在PyFunctionObject中与字节码指令捆绑在一起,所以PyFunctionObject也是一个Python中闭包的具体表现。
闭包是最内嵌套的一种实现。
不用闭包我们也能实现上面相同的功能,

上面这个就不是闭包,为什么呢?
因为最后一个print语句中传入的base=1的情况,居然能够改变基准值,这里如果是闭包的话,那就会抛出异常的。
我们利用函数默认值,居然实现了闭包的效果。
那么,闭包和默认参数的实现方式是不是相似的呢?
1.1实现闭包的基石
闭包的创建通常是利用嵌套函数来完成的。在PyCodeObject中,与嵌套函数相关的属性有co_cellvars 、co_freevars。前者是一个保存嵌套的作用域中使用的变量的集合,后者是保存使用了外层作用区域中的变量的集合,两者均为tuple。
PyFrameObject对象中,也有与闭包实现相关的属性,这就是f_localsplus,在PyFrame_New中,extras = code->co_stacksize + code->co_nlocals+ncells+ nfrees;
extras正是f_localsplus指向的那片内存的大小。
PyFunctionObject中,还有一个与闭包实现有关的属性。
1.2闭包实现
1.2.1创建closure
在PyCodeObject的co_cellvars中有东西,在PyEval_EvalCodeEx中,Python虚拟机就会如同处理默认参数一样,将co_cellvars中的东西拷贝到新创建的PyFrameObject的f_localsplus中。
python虚拟机先获得被内层嵌套函数引用的符号名,然后就创建一个cell对象,随后cell对象被拷贝到新创建的PyFrameObject的f_localsplus中。位置为co->co_nlocals+i,说明在f_localsplus中,cell对象的位置是在局部变量之后的。
上面总结就是:把PyCodeObject中的co_cellvars写道PyFrameObject的f_localsplus.
处理了cell对象之后,Python虚拟机进入PyEval_EvalFrameEx,从而正式开始函数调用。
从PyFrame中取得f_localsplus中的cell对象,存为freevars对象。在PyFunctionObject中,存储需要传递的内容。
2.2使用closure
闭包是在外层函数中创建,在内层函数中使用。
在将内层函数赋值给一个名字之后,例如show_value,
然后在show_vale()的时候,发现内层函数对应的PyCodeObject中的co_flags中包含了CO_NESTED,所以会进入到PyEval_EvalCodeEx.
而在内层函数对应的PyCodeObject中co_freevars里有引用外层函数中的符号名字,在PyEval_EvalCodeEx中,就会对这个co_freevars进行处理。
即在PyFunctionObject对象中与PyCodeObject对象绑定的装满了PyCellObject对象的tuple,所以这里其实偶是将PyCellObject对象一个一个地放入到f_localsplus中相应的位置。
处理完之后,PyFrameObject中就有闭包需要的元素了。
这里和调用外层函数时一致的,在内层函数调用的过程中,当引用外层作用域符号时,一定是到f_localsplus中的free变量区域中获取得到符号对应的值。
到这里我们已经讲完closure的创建到传递到使用的全过程了。
2.3 decorator
装饰器的本质其实就是闭包。
其他没啥好说的了,只是包装了一下而已。
python嵌套函数、闭包与decorator的更多相关文章
- Python 嵌套函数和闭包
Python 嵌套函数和闭包 1.函数嵌套 如果在一个函数内部定义了另一个函数,我们称外部的函数为外函数,内部的函数为内函数,如下代码: def out_func(): def inner_func1 ...
- python之函数闭包、可迭代对象和迭代器
一.函数名的应用 # 1,函数名就是函数的内存地址,而函数名()则是运行这个函数. def func(): return print(func) # 返回一个地址 # 2,函数名可以作为变量. def ...
- python 嵌套作用域 闭包函数
#闭包函数 def multiplier(factor): def multiplyByFactory(number): return number*factor return multiplyByF ...
- 【python】函数闭包
列表时可以改
- Python中函数的嵌套及闭包
函数的嵌套 调用:在函数中调用函数 定义:在函数中定义函数 地址:函数名有内存地址,内存地址可赋值 示例 a = 1 def outer(): a = 1 def inner(): a = 2 def ...
- ~~函数基础(三):嵌套函数&匿名函数~~
进击のpython 嵌套函数&匿名函数 讲完作用域之后 对变量的作用范围有大致的了解了吗? 讲个稍微小进阶的东西吧 能够帮助你更加的理解全局和局部变量 嵌套函数 玩过俄罗斯套娃不? 没玩过听过 ...
- 一文搞懂Python函数(匿名函数、嵌套函数、闭包、装饰器)!
Python函数定义.匿名函数.嵌套函数.闭包.装饰器 目录 Python函数定义.匿名函数.嵌套函数.闭包.装饰器 函数核心理解 1. 函数定义 2. 嵌套函数 2.1 作用 2.2 函数变量作用域 ...
- python基础—函数嵌套与闭包
python基础-函数嵌套与闭包 1.名称空间与作用域 1 名称空间分为: 1 内置名称空间 内置在解释器中的名称 2 全局名称空间 顶头写的名称 3 局部名称空间 2 找一个名称的查找顺序: ...
- Python之函数对象、函数嵌套、名称空间与作用域、闭包函数、装饰器
目录 一 函数对象 二 函数的嵌套 三 名称空间和作用域 四 闭合函数 五 装饰器 一.函数对象 1.函数是第一类对象 #第一类对象的特性:# 可以被引用 # 可以当做参数传递 # 返回值是函数 # ...
随机推荐
- Android 从网络中获取数据时 产生部分数据乱码的解决
产生部分数据乱码的解决 标签: android部分中文乱码 2014-04-12 23:24 12366人阅读 评论(10) 收藏 举报 分类: [Android 基础](15) 版权声明:本文为博主 ...
- apache 多站点配置
安装时可以选择gbk-gb2312的编码(我自己用utf-8). 2: 修改本机的hosts文件,如下: 示例:127.0.0.1 localhost127.0.0.1 ...
- Java开发环境的搭建以及使用eclipse从头一步步创建java项目
一.java 开发环境的搭建 这里主要说的是在windows 环境下怎么配置环境. 1.首先安装JDK java的sdk简称JDK ,去其官方网站下载最近的JDK即可..http://www.orac ...
- AngularJS 之 Factory vs Service vs Provider【转】
英文原文:AngularJS: Factory vs Service vs Provider 当你初试 Angular 时,很自然地就会往 controller 和 scope 里堆满不必要的逻辑.一 ...
- Golang gzip的压缩和解压
package src import ( "bytes" "compress/gzip" ) func GzipEncode(in []byte) ([]byt ...
- Jquery遮罩插件,想罩哪就罩哪!
一 前言 在项目开发时发现没有一个用起来 爽一点的遮罩插件,看起来觉得不难 好吧那就利用空闲时间,自己折腾一个吧,也好把jquery再温习一下, 需要的功能如下 1 可以全屏遮 用于提交数据时 2 ...
- Windows phone应用开发[20]-禁止Pivot手势
相比Ios 和Android. Windows Phone 控件库中多了两个比较特殊的空间Pivot 枢轴和Panamera 全景视图控件.在基于枢轴控件Pivot中我们经常会碰到一些比较特殊应用场景 ...
- RapidJSON 代码剖析(四):优化 Grisu
我曾经在知乎的一个答案里谈及到 V8 引擎里实现了 Grisu 算法,我先引用该文的内容简单介绍 Grisu.然后,再谈及 RapidJSON 对它做了的几个底层优化. (配图中的<Grisù& ...
- docfx开源啦
废话不多说了,直接上地址: 源代码: https://github.com/dotnet/docfx 文档: http://dotnet.github.io/docfx/ clone git clon ...
- 【BZOJ 4561】【JLOI 2016】圆的异或并
http://www.lydsy.com/JudgeOnline/problem.php?id=4561 一开始并不会做,后来看题解看懂了. 看懂了之后还是错了好几次,数组大小手残开小了. 圆的包含并 ...