Python 中的闭包与装饰器的详解

闭包(closure)是函数式编程的重要的语法结构。闭包也是一种组织代码的结构,它同样提高了代码的可重复使用性。

如果在一个内嵌函数里,对在外部函数内(但不是在全局作用域)的变量进行引用,那么内嵌函数就被认为是闭包(closure)。

定义在外部函数内但由内部函数引用或者使用的变量称为自由变量。

总结一下,创建一个闭包必须满足以下几点:

1. 必须有一个内嵌函数
2. 内嵌函数必须引用外部函数中的变量
3. 外部函数的返回值必须是内嵌函数
###1.闭包使用示例

先看一个闭包的例子:

    In [10]: def func(name):
        ...:     def in_func(age):
        ...:         print 'name:',name,'age:',age
        ...:     return in_func
        ...:
   
    In [11]: demo = func('feiyu')
   
    In [12]: demo(19)
    name: feiyu age: 19
这里当调用 func 的时候就产生了一个闭包——in_func,并且该闭包持有自由变量——name,因此这也意味着,当函数func的生命周期结束之后,name这个变量依然存在,因为它被闭包引用了,所以不会被回收。

在 python 的函数内,可以直接引用外部变量,但不能改写外部变量,因此如果在闭包中直接改写父函数的变量,就会发生错误。看以下示例:

实现一个计数闭包的例子:

    def counter(start=0):
        count = [start]
        def incr():
            count[0] += 1
            return count
        return incr
    
    a = counter()
    print 'a:',a
 
    In [32]: def counter(start=0):
        ...:     count = start
        ...:     def incr():
        ...:         count += 1
        ...:         return count
        ...:     return incr
        ...:
   
    In [33]: a = counter()
   
    In [35]: a()  #此处会报错
   
    UnboundLocalError: local variable 'count' referenced before assignment
应该像下面这样使用:

    In [36]: def counter(start=0):
        ...:     count = [start]
        ...:     def incr():
        ...:         count[0] += 1
        ...:         return count
        ...:     return incr
        ...:
   
    In [37]: count = counter(5)
   
    In [38]: for i in range(10):
        ...:     print count(),
        ...:    
    [6] [7] [8] [9] [10] [11] [12] [13] [14] [15]
###2.使用闭包的陷阱

    In [1]: def create():
       ...:     return [lambda x:i*x for i in range(5)]  #推导式生成一个匿名函数的列表
       ...:
   
    In [2]: create()
    Out[2]:
    [<function __main__.<lambda>>,
     <function __main__.<lambda>>,
     <function __main__.<lambda>>,
     <function __main__.<lambda>>,
     <function __main__.<lambda>>]
   
    In [4]: for mul in create():
       ...:     print mul(2)
       ...:    
    8
    8
    8
    8
    8
结果是不是很奇怪,这算是闭包使用中的一个陷阱吧!来看看为什么?

在上面的代码当中,函数create返回一个list里面保存了4个函数变量,这4个函数都共同的引用了循环变量i, 也就是说它们共享着同一个变量i,i是会改变的,当函数调用时,循环变量i已经是等于4了,因此4个函数返回的都是8。如果,需要在闭包使用循环变量的值的话,把循环变量作为闭包的默认参数或者是通过偏函数来实现。实现的原理也很简单,就是当把循环变量当参数传入函数时,会申请新的内存。示例代码如下:

    In [5]: def create():
       ...:         return [lambda x,i=i:i*x for i in range(5)]
       ...:
    In [7]: for mul in create():
       ...:     print mul(2)
       ...:    
    0
    2
    4
    6
    8
3,闭包与装饰器
装饰器就是一种的闭包的应用,只不过其传递的是函数:

   
    def addb(func):
        def wrapper():
            return '<b>' + func() + '</b>'
        return wrapper
   
    def addli(func):
        def wrapper():
            return '<li>' + func() + '</li>'
        return wrapper
   
    @addb         # 等同于 demo = addb(addli(demo))
    @addli        # 等同于 demo = addli(demo)
    def demo():
        return 'hello world'
   
    print demo()    # 执行的是 addb(addku(demo))
在执行时,首先将demo函数传递给addli进行装饰,然后将装饰后的函数传递给addb进行装饰。所以最后返回的结果是:

    <b><li>hello world</li></b>
4.装饰器中的陷阱
当你写了一个装饰器作用在某个函数上,这个函数的重要的元信息比如名字、文档字符串、注解和参数签名都会丢失。

    def out_func(func):
        def wrapper():
            func()
        return wrapper
   
    @out_func
    def demo():
        """
            this is  a demo.
        """
        print 'hello world.'
   
    if __name__ == '__main__':
        demo()
        print "__name__:",demo.__name__
        print "__doc__:",demo.__doc__
看结果:

    hello world.
    __name__: wrapper
    __doc__: None
函数名字和文档字符串都变成了闭包的信息。好在可以使用 functools 库中的 @wraps 装饰器来注解底层包装函数。

    from functools import wraps
   
    def out_func(func):
        @wraps(func)
        def wrapper():
            func()
        return wrapper
自己试试结果吧!

时间: 2024-10-29 03:35:04

Python 中的闭包与装饰器的详解的相关文章

Python中的装饰器用法详解

来源:http://www.jb51.net/article/59867.htm 来源:http://blog.csdn.net/mdl13412/article/details/22608283 这篇文章主要介绍了Python中的装饰器用法,以实例形式详细的分析了Python中的装饰器的使用技巧及相关注意事项,具有一定参考借鉴价值,需要的朋友可以参考下 本文实例讲述了Python中的装饰器用法.分享给大家供大家参考.具体分析如下: 这里还是先由stackoverflow上面的一个问题引起吧,如

Python中使用不同编码读写txt文件详解

  这篇文章主要介绍了Python中使用不同编码读写txt文件详解,本文给出不同编码下的读写文件代码方法,需要的朋友可以参考下 代码如下: import os import codecs filenames=os.listdir(os.getcwd()) out=file("name.txt","w") for filename in filenames: out.write(filename.decode("gb2312").encode(&q

Python中Continue语句的用法的举例详解

  这篇文章主要介绍了Python中Continue语句的用法的举例详解,是Python入门中的基础知识,需要的朋友可以参考下 Python continue语句返回while循环的开始.Continue语句拒绝在该循环的当前迭代中的其余语句执行并移动控制返回到循环的顶部(开始位置). continue语句可以在while和for循环使用. 语法 Python continue语句的语法如下: continue 流程图: 例子 ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14

Python 进阶_闭包 &amp;amp; 装饰器

目录 目录 闭包 函数的实质和属性 闭包有什么好处 小结 装饰器 更加深入的看看装饰器的执行过程 带参数的装饰器 装饰器的叠加 小结 装饰器能解决什么问题 小结 闭包 Closure: 如果内层函数引用了外层函数的局部变量(L),并且在外层函数中 return 内层函数时,这种关系就称之为闭包. 闭包的特点就是返回的内层函数还引用了外层函数的局部变量,所以要想正确的使用闭包,那么就要确保这个被内层函数引用的局部变量是不变的. EXAMPLE: In [71]: def count(): ...:

Python中的装饰器用法详解_python

本文实例讲述了Python中的装饰器用法.分享给大家供大家参考.具体分析如下: 这里还是先由stackoverflow上面的一个问题引起吧,如果使用如下的代码: 复制代码 代码如下: @makebold @makeitalic def say():    return "Hello" 打印出如下的输出: <b><i>Hello<i></b> 你会怎么做?最后给出的答案是: 复制代码 代码如下: def makebold(fn):    

Python中字符串的修改及传参详解_python

发现问题 最近在面试的时候遇到一个题目,选择用JavaScript或者Python实现字符串反转,我选择了Python,然后写出了代码(错误的): #!/usr/bin/env python #-*-coding:utf-8-*- __author__ = 'ZhangHe' def reverse(s): l = 0 r = len(s) - 1 while l < r: s[l],s[r] = s[r],s[l] l += 1 r -= 1 return s 然后面试官问了两个问题: (1)

Python中类型关系和继承关系实例详解

  本文详细介绍了Python中类型关系和继承关系.分享给大家供大家参考.具体分析如下: 如果一个对象A持有另一个对象B的ID,那么检索到A之后就可以检索到B,我们就说存在一个A到B的导航.这种导航关系使得Python中所有对象之间形成了一个复杂的网络结构. Python程序的运行包括: 1. 修改这个网络结构; 2. 执行有副作用的代码对象(code object或者说bytecode,见Python Language Reference 3.2) (副作用是指影响Python虚拟机之外的设备

Java 装饰器模式详解

转载请注明出处:http://blog.csdn.net/zhaoyanjun6/article/details/56488020 前言 在上面的几篇文章中,着重介绍了java 中常见的 IO 相关知识,在学习的过程中,发现 IO 包中是用了大量的装饰器模式,为了彻底的学习 IO,今天就来揭开装饰器模式的面纱. 为了弄明白装饰器模式的本质,我查看了很多资料,发现有很多文章要么说的很苦涩,要么举的例子不恰当. 其实我们可以这样理解装饰器模式, 就拿自己举例子,你把自己裸体的样子,想象成被装饰的对象

Python中的进程分支fork和exec详解_python

在python中,任务并发一种方式是通过进程分支来实现的.在linux系统在,通过fork()方法来实现进程分支. 1.fork()调用后会创建一个新的子进程,这个子进程是原父进程的副本.子进程可以独立父进程外运行. 2.fork()是一个很特殊的方法,一次调用,两次返回. 3.fork()它会返回2个值,一个值为0,表示在子进程返回;另外一个值为非0,表示在父进程中返回子进程ID. 以下只能在linux中运行,不能在window下运行. 进程分支fork() 实例如下: 复制代码 代码如下: