20 November 2013

之前经常听人提到“闭包”这个词,最近一边工作一边学习Python,看到Python中的闭包,结合《Python Cookbook》中的对闭包的介绍真正领悟了闭包的奥秘。

下面先举个例子:

def make_adder(addend):
    def adder(augend):
        return augend + addend
    return adder

make_adder函数里嵌套了一个内层函数adder,这个内层函数就是一个闭包,其实可以也不用管这个“闭包”的概念,先来看下这种模式解决些什么问题,“闭包”只是个名称罢了。

调用make_adder函数:p = make_adder(23),由于内层的函数adder里的逻辑用到了make_adder函数的入参,而这时这个入参addend绑定了值23,由于make_adder函数返回的是函数adder,所以这时的p其实就是内部的addend绑定了23的函数adder;同理可知,q = make_adder(44),这里的q就是内部addend绑定了44的函数adder,p和q这两个adder函数是不相同的,因为内部addend绑定的值不同,只是两个函数的模板相同罢了,这时我们执行p(100)得到结果是123,q(100)得到结果是144。这样做有什么用呢?其实可以这样看:我们可以把p = make_adder(23)和q = make_adder(44)看成是配置过程,23和44是配置信息,p(100)和q(100)根据不同的配置获得不同的结果,这样我们就可以解决开发中“根据配置信息不同获得不同结果”的问题

就拿《Python Cookbook》中的例子来阐述

我们需要一个字符串过滤的功能,把一个字符串中我们想留下的字符留下,除此之外的字符都过滤掉,这里的“我们想留下的字符”其实就是上面提到的”配置信息”,我们根据不同的配置信息获得不同的结果,我们根据我们想留下的字符得到不同的过滤后的字符串,所以这里的过滤功能函数这样写:

import string
allchars = string.maketrans('', '')
def makefilter(keep):
	delchars = allchars.translate(allchars, keep)
	def thefilter(s):
	    return s.translate(allchars, delchars)
	return thefilter

这里用到了python的translate,不清楚的可以看http://docs.python.org/

makefilter函数的参数是我们想留下的字符,可以传一个字符串,相当于字符串里所有的字符都要留下,makefilter函数里有个内层函数thefilter,这个函数会得到我们传入makefilter函数的想要留下的字符,根据delchars它就知道要过滤掉哪些字符,下面我们来执行下,看下效果:

>>> just_vowels = makefilter('aeiouy')      #我们想留下aeiouy这些字符
>>> just_vowels('wwwwwwwaeiouy')            #这里的 just_vowels其实就是知道了我们想留下的字符aeiouy后的thefilter函数 最后得到的结果:'aeiouy'

以上就是闭包的奥秘,其实就是一个“根据不同配置信息得到不同结果”的功能,而且我们在开发中是经常碰到这样的需求的,用闭包来解决真的非常好,代码简单易懂,而且扩展性也非常好!

###参考 1