Python闭包的学习

Python闭包的学习

什么是闭包?

借用维基上解释:在计算机科学中,闭包(Closure)是词法闭包(Lexical Closure)的简称,是引用了自由变量的函数。这个被引用的自由变量将和这个函数一同存在,即使已经离开了创造它的环境也不例外。所以,有另一种说法认为闭包是由函数和与其相关的引用环境组合而成的实体。闭包在运行时可以有多个实例,不同的引用环境和相同的函数组合可以产生不同的实例。

好吧,看了这段定义的确不是还不能立即理解闭包到底是什么。不过,闭包并不是很难理解,往下看几个小例子就能明确它到底是什么了。

前提介绍:

在了解闭包前,我们需要了解python语法中变量的作用范围(对理解后面的闭包的例子有帮助)。Python从语法上支持多个函数嵌套。在python现有的(2.1版本后)嵌套规则下,嵌套函数可以访问全局作用域自己本身的作用域以及自己之上的函数的局部作用域。如(来自python核心编程一书):

变量范围例一:

x = 1
def foo():
	y = 2
	def bar():
		z = 3
		print x + y + z
	bar()
foo()

结果为输出:

6

解释:

也就是bar()函数能够访问全局变量x,自己的局部作用域z,并且还能访问嵌套自己的foo()函数的局部变量y。

接着,我们还需要知道在python2.x中,里层函数虽然能够访问外层函数的变量,但是却不能对其赋值。我们对例一稍作修改:

变量范围例二:

x = 1
def foo():
	y = 2
	def bar():
		z = 3
		y += 1
		print x + y + z
	bar()
foo()

结果:

此时运行将报错:UnboundLocalError: local variable ‘y‘ referenced before assignment。

这个错误的意思是说本地变量y在引用前没有被定义。这是因为在里层函数里对变量进行赋值时,python将认为这个变量是本地的,所以如果有修改变量的值就会变成局部变量(当然如果没有修改,那么python将先在自己定义的范围内查找,如果没有则去外层查找,这样就知道了y是个外层的变量)。

这在python 3.x中可以使用nonlocal来解决,nonlocal关键字是用来在函数或其他作用域中使用外层(非全局)变量的。

了解了python中的变量作用域后,我们来看几个闭包的例子(来自python核心编程一书):

闭包例一:

def counter(start_at=0):
	count = [start_at]
	def incr():
		count[0] += 1
		return count[0]
	return incr

结果:

>>> count = counter(5)
>>> print count()
6
>>> print count()
7
>>> count2 = counter(100)
>>> print count2()
101

解释:

在counter函数中,我们定义了一个incr函数,incr函数的作用就是将count[0]的值加一,counter函数返回incr函数。注意,要知道,python中一切阶对象,函数也是可以作为函数对象进行传递和返回的。

当我们执行count = counter(5)时,count就被赋予了incr这个函数对象,也就说可以调用count就相当于调用incr函数。另外,我们给counter函数一个参数5,那么count[0]的初始值就为5了,把这个返回来的给定了参数的函数对象赋值给了count,那么每次调用count就会执行count[0]加一,并且返回count[0]的值。

综上,我们可以看出,counter()函数通过不同的参数,得到的函数对象是不一样的,也就是说,我们能够通过这种形式将函数和一个参数绑定在一起赋值给一个对象,然后再通过这个对象进行相关的操作。

注意:这里不能将counter()中的count由列表换成单一变量,否则会报UnboundLocalError错误(前提介绍有说明为什么),但是列表可以。

闭包例二:

def xy_add(x):
	def adding(y):
		#根据前面的函数嵌套变量作用域介绍,你应该知道adding()中可以访问x
		return x + y
	return adding

结果:

>>> myadd = xy_add(3)
>>> print myadd(4)
7
>>> print myadd(3)
6

解释:

xy_add()函数将范围其嵌套的adding函数对象,当我们给定xy_add()函数参数时,比如给定xi了,并将这个参数下返回的函数对象赋值给myadd后,myadd就是一个带有特定参数的函数,通过调用myadd(yi),我们就能得到任何一个xi与yi的相加之和。

意义和作用:

说实话,我初学python,还没怎么用过闭包,javascript中倒是用过,不过印象已经不深了。这里摘录在网上其他人的看法。

1>    函数和对象的根本目的是以某种逻辑方式组织代码,并提高代码的可重复使用性。闭包也是一种组织代码的结构,它同样提高的代码的可重复使用性。出处

2>    当闭包执行完后,仍然能够保持住当前的运行环境。比如控制棋子运动过程中,闭包可以利用上一次的棋子坐标进行运算。出处

3>    闭包可以根据外部作用域的局部变量来得到不同的结果,这有点像一种类似配置功能的作用,我们可以修改外部的变量,闭包根据这个变量展现出不同的功能。比如有时我们需要对某些文件的特殊行进行分析,先要提取出这些特殊行。出处

4>    利于并行运算,类比装饰器作用等等。

时间: 2024-10-09 19:22:43

Python闭包的学习的相关文章

python函数基础学习

函数的定义与调用: def 函数名(参数1,参数2): '''函数注释''' print('函数体') return 返回值 定  义:def关键字开关,空格之后接函数名和圆括号,最后冒号结尾 def 是固定且不可变的 函数名:函数名是包含字母.数字.下划线的任意组合,(不能以数字开头) 函数调用: 返回值 = 函数名 (参数1,参数2) 函数返回值: 1- return的作用:结束一个函数的执行 2- 首先返回值可以是任何的数据类型 3- 函数可以有返回值:如有返回值,必须要用变量接受才有效果

【python标准库学习】thread,threading(二)多线程同步

继上一篇介绍了python的多线程和基本用法.也说到了python中多线程中的同步锁,这篇就来看看python中的多线程同步问题. 有时候很多个线程同时对一个资源进行修改,这个时候就容易发生错误,看看这个最简单的程序: import thread, time count = 0 def addCount(): global count for i in range(100000): count += 1 for i in range(10): thread.start_new_thread(ad

【python标准库学习】thread,threading(一)多线程的介绍和使用

在单个程序中我们经常用多线程来处理不同的工作,尤其是有的工作需要等,那么我们会新建一个线程去等然后执行某些操作,当做完事后线程退出被回收.当一个程序运行时,就会有一个进程被系统所创建,同时也会有一个线程运行,这个线程就是主线程main,在主线程中所创建的新的线程都是子线程,子线程通常都是做一些辅助的事.python中提供了thread和threading两个模块来支持多线程. python中使用线程有两种方式,第一种是用thread模块的start_new_thread函数,另一种是用threa

python第三方库学习之xlrd读取Excel文件

因为经常会涉及到从Excel表中导数据,所以就学习了python的xlrd来读取excel中的数据. 1.xlrd的安装 xlrd是python的第三方库,所以是需要自己安装的,可以在python的官网http://pypi.python.org/pypi/xlrd下载该模块来安装,也可以通过其他手段,比如easy_install或者pip啥的,我已经安装好pip所以就用最懒的方式来安装了pip install xlrd来安装. 2.分析excel文件的层级对象 要读取excel的数据,就要了解

《灰帽Python-黑客和逆向工程师的Python编程》学习记录

ctypes是Python语言的一个外部库,提供和C语言兼容的数据类型,可以很方便的调用C DLL中的函数. 操作环境:CentOS6.5 Python版本:2.66 ctypes是强大的,强大到本书以后介绍的几乎所有库都要基于此.使用它我们就能够调用动态链接库中函数,同时创建各种复杂的C数据类型和底层操作函数.毫无疑问,ctypes就是本书的基础. 第一章 搭建开发环境 1.3.2 使用动态链接库 from ctypes import * libc = CDLL("libc.so.6"

python源码学习(一)——python的总体架构

python源码学习(一)——python的总体架构 学习环境: 系统:ubuntu 12.04 STLpython版本:2.7既然要学习python的源码,首先我们要在电脑上安装python并且下载python的源码,ubuntu 12.04 STL自带的python版本为2.7.3,这就可以了,接下来下载源码,下载python2.7.7的源代码,下载结束后解压文件,我们会看到如下文件结构:,接下来介绍一下,demo文件夹里放的是一些例子:Doc文件夹里放的是文档,Grammer是语法分析器,

Python结合OpenCV学习资料

原文地址:http://leihuang.net/2014/05/19/List-Interviews/ 单链表的一些常见面试题汇总 单链表反转/逆序 求单链表倒数第N个数 找到单链表的中间结点 如何判断链表是否有环的存在 单链表建环,无环链表变有环 如何知道环的长度? 如何找出环的连接点在哪里? 删除单链表中的重复元素 下面我先简单叙述一下每道题的思路,然后把实现的程序一起贴出来,不会讲得太细,我觉得只要有了思路之后,接下来的难点就是语言上的一些细节问题了,这个不自己去实现,听别人讲是体会不到

SecureCRT中python脚本编写学习指南

SecureCRT中python脚本编写学习指南 SecureCRT python 引言 在测试网络设备中,通常使用脚本对设备端进行配置和测试以及维护:对于PE设备的测试维护人员来说使用较多是SecureCRT工具:SecureCRT支持VB.JavaScript.Python等多种脚本语言,为了实现脚本在CRT中更加丰富稳定地执行,掌握CRT的常用函数是非常有用的.接下来的时间我将对SecureCRT脚本编写的常用函数展开学习应用. 内容 (1)使用python语言实现SecureCRT中的D

Python闭包与函数对象

1. Python闭包是什么 在python中有函数闭包的概念,这个概念是什么意思呢,查看Wikipedia的说明如下: “ In programming languages, closures (also lexical closures or function closures) are a technique for implementing lexically scoped name binding in languages with first-class functions. Ope