Python 进阶_闭包 & 装饰器

目录

  • 目录
  • 闭包
    • 函数的实质和属性
    • 闭包有什么好处
    • 小结
  • 装饰器
    • 更加深入的看看装饰器的执行过程
    • 带参数的装饰器
    • 装饰器的叠加
    • 小结
    • 装饰器能解决什么问题
      • 小结

闭包

Closure: 如果内层函数引用了外层函数的局部变量(L),并且在外层函数中 return 内层函数时,这种关系就称之为闭包。
闭包的特点就是返回的内层函数还引用了外层函数的局部变量,所以要想正确的使用闭包,那么就要确保这个被内层函数引用的局部变量是不变的。
EXAMPLE:

In [71]: def count():
    ...:     fs = []
    ...:     for i in range(1, 4):
    ...:         def f():
    ...:              return i*i
    ...:         fs.append(f)
    ...:     return fs
    ...:
    ...: f1, f2, f3 = count()
    ...:

In [72]: f1,f2,f3
Out[72]: (<function __main__.f>, <function __main__.f>, <function __main__.f>)

In [73]: f1(),f2(),f3()
Out[73]: (9, 9, 9)

最总 f1(),f2(),f3() 输出的是 (9,9,9) 而不是 (1,4,9)
原因就是当 count() 函数返回了 f1(),f2(),f3() 3个函数时,这3个函数所引用的变量 i 的值已经变成了 3。由于 f1、f2、f3 并没有被调用,所以,此时他们并未计算 i*i 。返回函数不要引用任何循环变量,或者后续会发生变化的变量。

函数的实质和属性

  • 函数是一个对象,在内存中会占用存储空间
  • 一般来说函数执行完成后其内部的变量都会被回收。但如果函数中的变量被 return 时,那么这个变量是不会被回收的,因为其引用计数 != 0
  • 函数具有属性
  • 函数具有返回值
  • 函数通过函数名来引用

EXAMPLE:

passline = 60
def func(val):
    print 'id(val): %x' % id(val)
    if val >= passline:
        print "pass"
    else:
        print "failed"

    def in_func():
        print val

    in_func()
    return in_func

f = func(89)
f()

print (f.__closure__)

Output:

In [33]: run demo_1.py
id(val): 33c6548     # 变量 val 在内存中的地址
pass
89
(<cell at 0x0000000004E22588: int object at 0x00000000033C6548>,)  # 函数 f 包含了一个 int 对象,这个 int 对象的引用跟变量 val 是相同的

NOTE1: in_func() 引用了嵌套作用域(E)中的变量 val
NOTE2: 并且内层函数 in_func() 被外层函数 func() return。 所以,val 这个变量被包含在了函数 in_func() 中一起被返回给了变量 f,即:变量 val 被变量 f 所引用,所以这个变量 val 并没有在函数 func() 执行结束之后被回收。

总结: 外层函数局部作用域中的变量能够被内部函数所引用。

闭包有什么好处?

因为闭包是通过 return 一个函数来定义的,所以我们可以把一些函数延迟执行。EXAMPLE:

def calc_sum(lst):
    def lazy_sum():
        return sum(lst)
    return lazy_sum

调用calc_sum()并没有计算出结果,而是返回函数:

>>> func = calc_sum([1, 2, 3, 4])
>>> func
<function lazy_sum at 0x1037bfaa0>

对返回的函数进行调用时,才计算出结果:

>>> func()
10

由于可以返回函数,所以我们在后续代码里就可以灵活的决定到底要不要调用该返回的函数。

再看下面一个例子:

def set_passline(passline):
    def cmp(val):
        if val >= passline:
            print "pass"
        else:
            print "failed"
    return cmp

f_100 = set_passline(60)
f_100(89)
print "f_100 type: %s" % type(f_100)
print "f_100 param: %s" % f_100.__closure__
print '*' * 20
f_150 = set_passline(90)
f_150(89)

Output:

In [37]: run demo_1.py
pass
f_100 type: <type 'function'>
f_100 param: <cell at 0x0000000004D5A468: int object at 0x00000000033C6038>
********************
failed

set_passline() 和 cmp() 组成一个闭包,最明显的好处在于可以提高 cmp() 函数内代码实现的复用性和封装(内层函数不能在全局作用域中被直接调用)。不需要为两个不用的判断(100 | 150)而写两个相似的函数。

而且外层函数中的局部变量还可以是一个函数。
继续看下一个例子:

def my_sum(*arg):
    print "my_sum"
    print arg
    return sum(arg)

def my_average(*arg):
    return sum(arg)/len(arg)

def dec(func):
    print "dec"
    print func
    def in_dec(*arg):
        print "in_dec"
        print func
        print arg
        if len(arg) == 0:
            return 0
        for val in arg:
            if not isinstance(val, int):
                return 0
        return func(*arg)
    return in_dec

# dec(my_sum)(1, 3, 4)
my_sum_result = dec(my_sum)
print my_sum_result(1, 3, 4)
print my_sum_result.__closure__

print '*' * 40

my_average_result = dec(my_average)
print my_average_result(1, 3, 4)
print my_average_result.__closure__

Output:

In [35]: run demo_1.py
dec
<function my_sum at 0x00000000048F1208>
in_dec
<function my_sum at 0x00000000048F1208>
(1, 3, 4)
my_sum
(1, 3, 4)
8
(<cell at 0x0000000004A7C618: function object at 0x00000000048F1208>,)
****************************************
dec
<function my_average at 0x00000000048F1A58>
in_dec
<function my_average at 0x00000000048F1A58>
(1, 3, 4)
2
(<cell at 0x0000000004A7CE58: function object at 0x00000000048F1A58>,)

小结

  • 闭包能够提高代码的复用和内层函数的封装
  • 内层函数引用外层函数的局部变量可以是基本数据类型对象,也可以是一个函数对象
  • 调用外层函数(传递外层形参)时,返回的是一个内存函数对象,然后可以再对内层函数传递形参,这样就增强了代码的灵活性。EG. 由外层设定条件,内层进行处理。

当然我们可以能回认为这样的写法并不美观,所以我们继续看 Python 的装饰器和闭包的关系。

装饰器

装饰器实际上就是一个函数, 而且是一个接收函数对象的函数. 有了装饰器我们可以在执行被装饰函数之前做一个预处理, 也可以在处理完函数之后做清除工作. 所以在装饰器函数中你会经常看见这样的代码: 函数体定义了一个内嵌函数, 并在内嵌函数内实现了目标函数(被装饰函数)的调用.这是一种 AOP 面向方法编程的方式.
- 装饰器用于装饰函数 ⇒ dec() 用于装饰 my_sum()
- 装饰器返回一个新的函数对象 ⇒ dec() 返回一个函数 in_dec()
- 被装饰的函数标识符指向返回的函数对象 ⇒ my_sum() 指向 in_dec() my_sum_result = dec(my_sum) 即: my_sum_resultdec() 返回的函数对象,引用指向了 my_sum()
- 语法糖:@装饰器名称

从上面几点特性可以看出,装饰器就是对闭包的使用。
将上面的例子转换一下写法:

def dec(func):
    print 'call dec()'
    def in_dec(*arg):
        print "call in_dec()"
        if len(arg) == 0:
            return 0
        for val in arg:
            if not isinstance(val, int):
                return 0
        return func(*arg)
    return in_dec

# @dec 执行的过程:
# 1. dec(my_sum) --> return in_dec
# 2. my_sum = in_dec
# 为了丰富被装饰函数的功能(这里是丰富了一个参数验证功能),是一种抽象的编程思维
@dec
def my_sum(*arg):
    print 'call my_sum()'
    return sum(arg)

def my_average(*arg):
    return sum(arg)/len(arg)

print my_sum(1,2,3,4,5)

Output:

In [43]: run demo_1.py
call dec()
call in_dec()
call my_sum()
15

更加深入的看看装饰器的执行过程

  • chocolate() 只是一个普通的函数
def chocolate():
    print "call chocolate()"
chocolate()

Output:

In [65]: run demo_1.py
call chocolate()

正常的调用了 chocolate() 函数

  • chocolate() 加上了一个没有返回内存函数的装饰器
def jmilkfan(func):
    def in_jmilkfan():
        print "call in_jmilkfan()"
        func()
    print "call jmilkfan()"

@jmilkfan
def chocolate():
    print "call chocolate()"
chocolate()

Output:

In [51]: run demo_1.py
call jmilkfan()  # 首先调用了 jmilkfan 装饰器对应的 jmilkfan() 函数

---------------------------------------------------------------------------
TypeError                                 Traceback (most recent call last)
D:\Development\Python27\workspace\demo_1.py in <module>()
      9     print "call chocolate()"
     10 #print 'typr chocolate:%s' % type(chocolate)
---> 11 chocolate()

TypeError: 'NoneType' object is not callable

**ERROR:**chocolate 是一个’NoneType’对象
因为 jmilkfan() 默认 return 了一个 None, 而不是内层函数 in_jmilkfan(),这里说明装饰器对应的 jmilkfan() 的确被调用了。但却没有返回内层函数给 chocolate 接收。

def jmilkfan(func):

    print 'func id:%x' % id(func)

    def in_jmilkfan():
        print "call in_jmilkfan()"
        func()

    print "call jmilkfan()"
    print 'in_jmilkfan():%s' % in_jmilkfan.__closure__
    return in_jmilkfan

@jmilkfan
def chocolate():
    print 'chocolate():%s' % chocolate.__closure__
    print "call chocolate()"
chocolate()

Output:

In [67]: run demo_1.py
func id:490ecf8
call jmilkfan()
in_jmilkfan():<cell at 0x0000000004B0F5B8: function object at 0x000000000490ECF8>
call in_jmilkfan()
chocolate():<cell at 0x0000000004B0F5B8: function object at 0x000000000490ECF8>
call chocolate()

可看出 choholate()、 in_jmilkfan()、 func 都是引用了同一个函数对象。即: in_jmilkfan() 被 chocolate() 接收了。

注意当被装饰的函数拥有形参时,装饰器的内层函数必须定义相同的形参,否则会报错。

def jmilkfan(func):

    print 'func id:%x' % id(func)

    def in_jmilkfan(x, y):
        print "call in_jmilkfan()"
        func(x, y)

    print "call jmilkfan()"
    print 'in_jmilkfan():%s' % in_jmilkfan.__closure__
    return in_jmilkfan

@jmilkfan
def chocolate(x, y):
    print 'chocolate():%s' % chocolate.__closure__
    print "call chocolate(), the value is:%d" % (x + y)

chocolate(1, 2)

Output:

In [70]: run demo_1.py
func id:490edd8
call jmilkfan()
in_jmilkfan():<cell at 0x0000000004AF3288: function object at 0x000000000490EDD8>
call in_jmilkfan()
chocolate():<cell at 0x0000000004AF3288: function object at 0x000000000490EDD8>
call chocolate(), the value is:3

带参数的装饰器

@decomaker(deco_args)
def foo():pass

# 等效于

foo = decomaker(deco_args)(foo)

带参数的装饰器和不带参数的装饰器的区别在于, 前者需要先传入一个参数并返回一个函数对象, 该函数对象才是实际作用于被装饰函数的装饰器.

装饰器的叠加

@deco1(deco_args)
@deco2
def func():pass

#等效于

func = deco1(deco_args)(deco2(func))

小结

装饰器的作用: 在提供了闭包的工作之外,优雅了语法

装饰器的使用过程

  • 调用了装饰器对应的外层函数,并且将被装饰函数作为实参传递给外层函数。目的是为了能够让内存函数使用这个作为外层函数的局部变量的被装饰函数,从而执行被装饰函数的代码实现。
  • 外层函数将内层函数的引用返回给被装饰函数,实现闭包。

装饰器能解决什么问题?

问题:如果我们定义了一个函数后,又想在运行时动态的为这个函数增加功能,但是又不希望改动这个函数本身的代码?那么我们能否将这个函数作为一个参数传递给另一个函数,从而产生一个拥有更多功能的新函数呢 ?

这就用到了所谓的高阶函数:
1. 可以接受函数作为参数(函数又函数名来引用,函数名的本质就是一个变量)
2. 可以返回一个函数
这样高阶函数就能够到达我们希望接收一个函数并为其进行包装后再返回一个新的函数的效果。
EXAMPLE:

func = new_func(func)
# new_func 是一个高阶函数,也可以成为一个装饰器
# 因为函数是通过函数名来引用的,所以我们可以将一个由高阶函数返回的新的函数再次赋值给原来的函数名
# 这样的话 func 的原始定义函数就被彻底隐藏起来了,达到了封装的效果
# Python 还可以通过 @ 来简化装饰器的调用, EG:
@new_func
def func(x):
    ...

小结

装饰器可以极大的简化代码,避免每个函数编写重复性的代码。
我们可以将需要被重复且希望动态调用的代码写成装饰器,EG:
引入日志(打印日志):@log
检测性能(增加计时逻辑来监测性能):@performance
加入事务能力(数据库事务):@transaction
URL 路由:@post('/register')

时间: 2024-09-15 16:48:05

Python 进阶_闭包 &amp; 装饰器的相关文章

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

闭包(closure)是函数式编程的重要的语法结构.闭包也是一种组织代码的结构,它同样提高了代码的可重复使用性. 如果在一个内嵌函数里,对在外部函数内(但不是在全局作用域)的变量进行引用,那么内嵌函数就被认为是闭包(closure). 定义在外部函数内但由内部函数引用或者使用的变量称为自由变量. 总结一下,创建一个闭包必须满足以下几点: 1. 必须有一个内嵌函数 2. 内嵌函数必须引用外部函数中的变量 3. 外部函数的返回值必须是内嵌函数 ###1.闭包使用示例 先看一个闭包的例子:     I

python闭包以及装饰器

通俗的定义:如果在一个内部函数里,对在外部作用域(但不是在全局作用域)的变量进行引用,那么内部函数就被认为是闭包(closure).它只不过是个"内层"的函数,由一个名字(变量)来指代,而这个名字(变量)对于"外层"包含它的函数而言,是本地变量; 1 #示例一: 2 #!/usr/bin/python 3 #encoding=utf-8 4 5 def add_a(num1): 6 print "num1:%d" % num1 7 def add

Python自动重试HTTP连接装饰器

  这篇文章主要介绍了Python自动重试HTTP连接装饰器,有时候我们要去别的接口取数据,可能因为网络原因偶尔失败,为了能自动重试,写了这么一个装饰器,可以实现自动重连2次,需要的朋友可以参考下 有时候我们要去别的接口取数据,可能因为网络原因偶尔失败,为了能自动重试,写了这么一个装饰器. 这个是python2.7x 的版本,python3.x可以用 nonlocal 来重写. ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 #-*- c

Python自动重试HTTP连接装饰器_python

有时候我们要去别的接口取数据,可能因为网络原因偶尔失败,为了能自动重试,写了这么一个装饰器. 这个是python2.7x 的版本,python3.x可以用 nonlocal 来重写. #-*- coding: utf-8 -*- #all decorators in this tool file #author: orangleliu ############################################################ #http连接有问题时候,自动重连 de

Python 进阶_函数式编程

目录 目录 函数式编程 Python 函数式编程的特点 高阶函数 匿名函数 lambda 函数式编程相关的内置函数 filter 序列对象过滤器 map reduce 折叠 自定义的排序函数 最后 函数式编程 首先要确定一点就是:函数 != 函数式,函数式编程是一种编程的范式. 特点: 把计算视为函数而非指令 纯函数式编程,不需要变量,没有副作用,测试简单 支持高阶函数,代码简洁 Python 函数式编程的特点 需要注意的是,Python 不是也不可能会成为一种纯函数是编程语言,但 Python

Python 进阶_模块 &amp;amp; 包

目录 目录 模块的搜索路径和路径搜索 搜索路径 命名空间和变量作用域的比较 变量名的查找覆盖 导入模块 import 语句 from-import 语句 扩展的 import 语句 as 自动载入模块 模块导入的特性 模块内建函数 __import__ globals locals reload Package 包 __init__py import package 模块的搜索路径和路径搜索 搜索路径 默认的模块搜索路径在 Python 解析器编译安装时被指定, 我们可以通过 sys 模块来查看

Python 进阶_迭代器 &amp;amp; 列表解析

目录 目录 迭代器 iter 内建的迭代器生成函数 迭代器在 for 循环中 迭代器与字典 迭代器与文件 创建迭代器对象 创建迭代对象并实现委托迭代 迭代器的多次迭代 列表解析 列表解析的样例 列表解析和迭代器 迭代器 迭代器是一个含有 next() 方法的对象,让我们可以迭代不是序列数据类型但表现出序列行为的对象,所以可以说迭代器为类序列对象提供了一个类序列的接口(只要是实现了 __iter__() 方法的对象,就可以使用迭代器来进行访问).迭代器从对象的第一个元素开始访问,直到所有的元素被遍

详解Python中的装饰器、闭包和functools的教程_python

装饰器(Decorators) 装饰器是这样一种设计模式:如果一个类希望添加其他类的一些功能,而不希望通过继承或是直接修改源代码实现,那么可以使用装饰器模式.简单来说Python中的装饰器就是指某些函数或其他可调用对象,以函数或类作为可选输入参数,然后返回函数或类的形式.通过这个在Python2.6版本中被新加入的特性可以用来实现装饰器设计模式. 顺便提一句,在继续阅读之前,如果你对Python中的闭包(Closure)概念不清楚,请查看本文结尾后的附录,如果没有闭包的相关概念,很难恰当的理解P

详解Python的装饰器

Python中的装饰器是你进入Python大门的一道坎,不管你跨不跨过去它都在那里. 为什么需要装饰器 我们假设你的程序实现了say_hello()和say_goodbye()两个函数. def say_hello():      print "hello!"        def say_goodbye():      print "hello!"  # bug here    if __name__ == '__main__':      say_hello()