什么是闭包?

闭包(closure)是词法闭包(lexical closure)的简称。闭包不是新奇的概念,而是早在高级程序语言开始发展的年代就已产生。

对闭包的理解大致分为两类,将闭包视为函数或者是由函数及其引用环境组成的复合体。

  1. 前者认为闭包是在其词法上下文中引用了自动变量的函数。
  2. 后者认为闭包是显示表示引用外部环境,并将它与函数捆绑在一起而产生的复合体。

所谓的自由变量是指局部变量以外的变量,同时自由变量的绑定可预知,所谓的绑定(binding)是指变量名称与其所代表的对象之间的联系。一些早期的计算机语言没有所谓封闭的函数,自由变量在运行时再进行绑定。对于那些拥有开放和封闭函数概念的计算机语言,人们需要一种新的术语来区分这两种函数,于是便有了术语闭包。

一般而言,第二种观点相对更为准确。因为函数是一些可执行的代码,这些代码在定义函数时被固化,在运行时不会发生任何变化,所以一个函数只有一个实例。而闭包会随着引用环境变化而产生不同的实例。

在 Python 以及大多数现代计算机语言中,所有函数都是封闭的,也就是说没有自由变量。于是对闭包的定义也从早期”对包含可预知绑定对象的自由变量的函数“变成了”可引用非激活环境的对象的函数“。对 Python 而言,自由变量被绑定到在外围作用域定义的变量。

Closure in Python

闭包(closure)是一种引用了外部变量的函数对象,无论该变量所处的作用域是否还存在于内存中。

举例来说,函数 generate_power_func 返回了另一个函数:

def generate_power_func(n):
print "id(n): %X" % id(n)
def nth_power(x):
return x**n
print "id(nth_power): %X" % id(nth_power)
return nth_power

函数 nth_power 就是一个闭包,它可以访问定义在 generate_power_func 函数中的变量 n,显而易见如果缺少变量 n,函数 nth_power 将是一个不能执行的没有闭合的函数,这个不完整的函数 nth_power 需要变量 n 来让它变成一个完整的函数对象,这种函数就是闭包,换句话说是变量 n 封闭了函数 nth_power

现在我们调用一下函数 generate_power_func,并将调用结果返回给一个变量 raised_to_4

>>> raised_to_4 = generate_power_func(4)
id(n): 28F8A4
id(nth_power): B6CB4570
>>> repr(rasied_to_4)
'<function nth_power at 0xb6cb4570>'

结果是调用函数 generate_power_func(4) 将生成一个 nth_power 函数对象。现在我们将函数 generate_power_func 函数从全局命名空间删除。

>>> del generate_power_func

然后调用闭包函数 raised_to_4

>>> rasied_to_4(2)
16

奇迹出现了,销毁函数 generate_power_func 并没有影响到函数 raised_to_4。我们在函数 generate_power_func 中定义变量 n,并且我们已经销毁了函数 generate_power_func,为什么 raised_to_4 会知道变量 n=4

这是因为函数对象 nth_power 是由 generate_power_func 产生的一个闭包,闭包会保留来自外围作用域变量的信息。

__closure__ 属性和 cell 对象

Python 中函数对象都拥有一个 __closure__ 属性。

__closure__ 对象返回一个由 cell 对象组成的元组,cell 对象记录了定义在外围作用域的变量信息。

>>> raised_to_4.__closure__
(<cell at 0xb6cb7f70: int object at 0x28f8a4>,)
>>> type(raised_to_4.__closure__[0])
<type 'cell'>
>>> raised_to_4.__closure__[0].cell_contents
4

正如所见,__closure__ 属性引用了一个 int 对象,这个 int 对象就是变量 n。对于那些不是闭包的函数对象来说,__closure__ 属性值为 None。

Ref:

Python Closures Explained

Closures in Python

闭包的概念、形式与应用

Python 闭包的更多相关文章

  1. Python闭包与函数对象

    1. Python闭包是什么 在python中有函数闭包的概念,这个概念是什么意思呢,查看Wikipedia的说明如下: “ In programming languages, closures (a ...

  2. Python闭包及装饰器

    Python闭包 先看一个例子: def outer(x): def inner(y): return x+y return innder add = outer(8) print add(6) 我们 ...

  3. Python闭包详解

    Python闭包详解 1 快速预览 以下是一段简单的闭包代码示例: def foo(): m=3 n=5 def bar(): a=4 return m+n+a return bar >> ...

  4. Python闭包及其作用域

    Python闭包及其作用域 关于Python作用域的知识在python作用域有相应的笔记,这个笔记是关于Python闭包及其作用域的详细的笔记 如果在一个内部函数里,对一个外部作用域(但不是全局作用域 ...

  5. Python 闭包小记

    闭包就是能够读取其他函数内部变量的函数.例如在javascript中,只有函数内部的子函数才能读取局部变量,所以闭包可以理解成“定义在一个函数内部的函数“.在本质上,闭包是将函数内部和函数外部连接起来 ...

  6. 理解Python闭包概念

    闭包并不只是一个python中的概念,在函数式编程语言中应用较为广泛.理解python中的闭包一方面是能够正确的使用闭包,另一方面可以好好体会和思考闭包的设计思想. 1.概念介绍 首先看一下维基上对闭 ...

  7. Python闭包举例

    Python闭包的条件: 1.函数嵌套.在外部函数内,定义内部函数. 2.参数传递.外部函数的局部变量,作为内部函数参数. 3.返回函数.外部函数的返回值,为内部函数. 举例如下: def line_ ...

  8. python 闭包和装饰器

    python 闭包和装饰器 一.闭包闭包:外部函数FunOut()里面包含一个内部函数FunIn(),并且外部函数返回内部函数的对象FunIn,内部函数存在对外部函数的变量的引用.那么这个内部函数Fu ...

  9. Python 闭包(Closure)

    Python  闭包 (Closure) 这里介绍一下python 的闭包 基本概念 闭包(closure)是函数式编程的重要的语法结构. 函数式编程的一个特点就是,允许把函数本身作为参数传入另一个函 ...

随机推荐

  1. javaweb学习总结(四)——Http协议

    一.什么是HTTP协议 HTTP是hypertext transfer protocol(超文本传输协议)的简写,它是TCP/IP协议的一个应用层协议,用于定义WEB浏览器与WEB服务器之间交换数据的 ...

  2. Extjs4中RadioGroup的赋值与取值

    1.定义rg var rg = new Ext.form.RadioGroup({ fieldLabel : "test", items : [{ boxLabel : '每天', ...

  3. iframe父子页面调用小结

     子页面调用父页面 $('#Id', window.parent.document); //调用父页面元素 window.parent.func1(); //调用父页面方法  (子页面同理,需将js方 ...

  4. cocos2d-x在Android平台下的音频导致的卡死

      先挖个坑,慢慢再来填.出错信息: frameworks/wilhelm/src/android/AudioPlayer_to_android.cpp:779: pthread 0x75f14a00 ...

  5. Python学习笔记(1):列表元组结构

    Python的列表元组功能强大,令人印象深刻.一是非常灵活,二是便于集体操作.特别是以元组作为列表项的结构,和数据访问的结果能够对应起来,和习惯的二维表理解上也一致,有很多的用途. 以学习笔记(3)中 ...

  6. Lucene 4.X 倒排索引原理与实现: (1) 词典的设计

    词典的格式设计 词典中所保存的信息主要是三部分: Term字符串 Term的统计信息,比如文档频率(Document Frequency) 倒排表的位置信息 其中Term字符串如何保存是一个很大的问题 ...

  7. zz Windows 10安装教程:硬盘安装Win10 系统步骤(适合32位和64位)

    Windows 10安装教程:硬盘安装Win10 系统步骤(适合32位和64位) Posted on 2015年01月28日 by 虾虾 22 Comments   最新的Windows 10 MSD ...

  8. LDO/DC-DC区别总结(转)

    电源是一个电子系统中不可缺少的非常重要的一部分.但是外接的电源通常不能够完全提供系统中需要的所有的电源种类.因此带来了电源电压的变换问题.常用的电源电压的变换芯片包括LDO和DC-DC两种.下面对这两 ...

  9. Generating a new ASP.NET session in the current HTTPContext

    void regenerateId() { System.Web.SessionState.SessionIDManager manager = new System.Web.SessionState ...

  10. Reset Entity-Framework Migrations

    You need to delete the state: Delete the migrations folder in your project Delete the __MigrationHis ...