python高级之装饰器

python高级之装饰器

本节内容

  1. 高阶函数
  2. 嵌套函数及闭包
  3. 装饰器
  4. 装饰器带参数
  5. 装饰器的嵌套
  6. functools.wraps模块
  7. 递归函数被装饰

1.高阶函数

高阶函数的定义:

满足下面两个条件之一的函数就是高阶函数:

  • 接受一个或多个函数作为输入参数
  • 输出一个函数

首先理解一个概念:函数名其实也是一个变量,一个函数其实就是一个对象,函数名就是对这个对象的引用。所以函数名也就和一个普通变量一样可以被当做函数的变量进行传递,当然也能够把函数名当做一个变量进行返回。

举个栗子:

 1 def foo(func,x,y):
 2     return func(x,y)
 3 def add(x,y):
 4     return x+y
 5 print(foo(add,3,4))
 6 #************************
 7 def foo():
 8     x=3
 9     def bar():
10         return x
11     return bar 
12 a=foo()
13 print(a())

上面的就是一些高阶函数。

2.嵌套函数及闭包

上面的例子里面的第二个例子,a=foo()的时候就是执行foo函数并且返回bar对象给a,这时候foo函数已经执行完了,a得到的是bar函数对应的内存地址。那么执行a()函数的时候为什么能够得到x的值呢?明明foo函数已经在前面执行完了,之后执行的是bar函数,但是bar函数能够获取到foo函数中定义的变量。

这种在内部嵌套函数里面能够获取到外部函数中的变量的现象在python里面叫做闭包,下面来说说闭包的定义:

定义:如果在一个内部函数里,对在外部作用域(但不是在全局作用域)的变量进行引用,那么内部函数就被认为是闭包(closure)。也可以叫做:闭包=函数块+定义函数时的环境。

3.装饰器

装饰器本质上是一个函数,该函数用来处理其他函数,它可以让其他函数在不需要修改代码的前提下增加额外的功能,装饰器的返回值也是一个函数对象。它经常用于有切面需求的场景,比如:插入日志、性能测试、事务处理、缓存、权限校验等应用场景。装饰器是解决这类问题的绝佳设计,有了装饰器,我们就可以抽离出大量与函数功能本身无关的雷同代码并继续重用。概括的讲,装饰器的作用就是为已经存在的对象添加额外的功能。

假如我们有一个之前已经写好了功能的函数,现在又想在上面添加计时功能,想统计这个函数执行花费了多少时间,但是又因为这个函数是一个基础函数,被很多其他地方调用了,所以不能修改这个函数中的代码。(这就是开发中的开放封闭原则:即写的代码对于需求来说是开放的,也就是说可以在原有需求上面添加需求,但是对于源代码来说又是封闭的,一个完成功能的函数,不能被修改。)这时候该怎么办呢?

举个栗子:

def add(*args,**kwargs):
    Sum = reduce(lambda x,y:x+y ,args)
    print(Sum)

这是一个计算加法的函数,那么怎么给这个做加法的函数统计时间呢?

这时,我们想到了可以使用另一个函数调用这个函数,再去统计这个函数执行的时间。

栗子如下:

 1 import time
 2 def add(*args,**kwargs):
 3         Sum = reduce(lambda x,y:x+y ,args)
 4         print(Sum)
 5
 6 def show_time(*args,**kwargs):
 7     start_time=time.time()
 8     add(*args,**kwargs)
 9     end_time=time.time()
10     print("计算时间:%s" % (end_time-start_time))
11
12 show_time()

这样,就能够统计到函数的执行时间了,但是。。。别人调用函数的时候就需要修改函数名了,那么该怎么办呢?这时候我们想到了闭包。将上面的函数进行改造下,成了下面的方式:

 1 import time
 2 def add(*args,**kwargs):
 3         Sum = reduce(lambda x,y:x+y ,args)
 4         print(Sum)
 5
 6 def show_time(func):
 7     def wrapper(*args,**kwargs)
 8         start_time=time.time()
 9         func(*args,**kwargs)
10         end_time=time.time()
11         print("计算时间:%s" % (end_time-start_time))
12     return wrapper
13
14 wrap=show_time(add)

这时候执行show_time函数之后返回的是什么呢?没错,返回的是一个wrapper函数对象,这时候执行这个函数对象,是不是就能够既能够获取到函数执行时间,又能够执行函数了呢?

下面我们再对上面的东西进行改造一下,效果如下:

 1 import time
 2
 3 def show_time(func):
 4     def wrapper(*args,**kwargs)
 5         start_time=time.time()
 6         func(*args,**kwargs)
 7         end_time=time.time()
 8         print("计算时间:%s" % (end_time-start_time))
 9     return wrapper
10
11 def add(*args,**kwargs):
12         Sum = reduce(lambda x,y:x+y ,args)
13         print(Sum)
14 add=show_time(add)
15 add(1,2,3,4,5)

这时候是不是又没有改变函数调用名称,又能够给这个函数加上一个计算时间的功能呢?

其实上面的例子就是装饰器的概念,只不过在python中使用@语法糖,可以将add=show_time(add)这一句隐士的表达出来罢了,下面是真正的实现一个简单的装饰器了。

 1 import time
 2
 3 def show_time(func):
 4     def wrapper(*args,**kwargs)
 5         start_time=time.time()
 6         func(*args,**kwargs)
 7         end_time=time.time()
 8         print("计算时间:%s" % (end_time-start_time))
 9     return wrapper
10 @show_time  # 其实这一句的意思就是:add=show_time(add)
11 def add(*args,**kwargs):
12         Sum = reduce(lambda x,y:x+y ,args)
13         print(Sum)
14
15 add(1,2,3,4,5)

上面就是python里面实现的一个简单的装饰器了。

4.装饰器带参数

既然,我们可以使用装饰器,对带参数的函数进行装饰,那么,我们能不能对装饰器带上参数,使装饰器更加灵活呢?

答案是可以的,那就是在装饰器的外面再嵌套一层函数。

现在在之前的需求上再增加一个需求,在装饰器中传递一个参数,实现可以通过这个参数决定是否将计算时间写入到文件中去,下面是实现的代码:

 1 import time
 2 def logger(flag=Falce):
 3     def show_time(func):
 4         def wrapper(*args,**kwargs)
 5             start_time=time.time()
 6             func(*args,**kwargs)
 7             end_time=time.time()
 8             print("计算时间:%s" % (end_time-start_time))
 9             if flag:
10                 print("写入日志文件")
11             else:
12                 print("不写入日志文件")
13         return wrapper
14     return showtime
15
16 @logger(flag=True)  # 首先会执行logger函数,返回show_time对象,然后@show_time就和之前的装饰器一样了
17 def add(*args,**kwargs):
18         Sum = reduce(lambda x,y:x+y ,args)
19         print(Sum)
20
21 add(1,2,3,4,5)

上面的logger函数接受参数,执行这个函数之后,返回的是一个装饰器对象,但是装饰器对象是一个闭包,可以拿到logger函数传递进来的参数。其内部的wrapper函数也能够拿到logger中传递进来的参数。

5.装饰器的嵌套

可不可以对装饰器进行多层嵌套呢?一层装饰器实现一个新功能,这样,想实现什么功能,就在外面再套一层装饰器罢了。

下面是两个装饰器嵌套的源代码:

 1 def tag2(func):  # 对返回的函数值进行第二次装饰
 2     def wrapper(*args,**kwargs):
 3         rst=func(*args,**kwargs)
 4         rst="".join(("<p>",rst,"</p>"))
 5         return rst
 6     return wrapper
 7
 8 def tag1(func):  # 对返回的函数值进行第一次装饰
 9     def wrapper(*args,**kwargs):
10         rst=func(*args,**kwargs)
11         rst="".join(("<a>",rst,"</a>"))
12         return rst
13     return wrapper
14
15 @tag2
16 @tag1
17 def helloworld():
18     return "hello world"
19
20 print(helloworld())

如上所示,定义了两个装饰器,tag1装饰器对helloworld函数进行装饰,在返回的helloworld字符串外面嵌套一层a标签,然后在tag2装饰器上,对前面使用第一个装饰器装饰过了的函数看成一个整体,再进行装饰,在最外面再嵌套一层p标签,这就是装饰器的嵌套。。。越上面的装饰器,表示越在外面,先执行里面的再执行外面的。

6.functools.wraps模块

使用装饰器极大地复用了代码,但是他有一个缺点就是原函数的元信息不见了,比如函数的docstring、name、参数列表,先看例子:

 1 def foo():
 2     print("hello foo")
 3
 4 print(foo.__name__)
 5 #####################
 6
 7 def logged(func):
 8     def wrapper(*args, **kwargs):
 9
10         print (func.__name__ + " was called")
11         return func(*args, **kwargs)
12
13     return wrapper
14
15
16 @logged
17 def cal(x):
18    return x + x * x
19
20
21 print(cal.__name__)
22
23 ########
24 # foo
25 # wrapper

解释:

 1 @logged
 2 def f(x):
 3    return x + x * x
 4 等价于:
 5
 6 def f(x):
 7     return x + x * x
 8 f = logged(f)
 9 不难发现,函数f被wrapper取代了,当然它的docstring,__name__就是变成了wrapper函数的信息了。
10
11 print f.__name__    # prints ‘wrapper‘
12 print f.__doc__     # prints None

这个问题就比较严重的,好在我们有functools.wraps,wraps本身也是一个装饰器,它能把原函数的元信息拷贝到装饰器函数中,这使得装饰器函数也有和原函数一样的元信息了。

 1 from functools import wraps
 2
 3 def logged(func):
 4
 5     @wraps(func)
 6
 7     def wrapper(*args, **kwargs):
 8         print (func.__name__ + " was called")
 9         return func(*args, **kwargs)
10     return wrapper
11
12 @logged
13 def cal(x):
14    return x + x * x
15
16 print(cal.__name__)  #cal

7.递归函数被装饰

在实际使用过程中发现,在递归函数中使用普通的装饰器得到的并不是我们想要的结果。

 1 ##----------------------------------------foo函数先加载到内存,然后foo变量指向新的引用,所以递归里的foo是wrapper函数对象
 2 # def show_time(func):
 3 #
 4 #     def wrapper(n):
 5 #         ret=func(n)
 6 #         print("hello,world")
 7 #         return ret
 8 #     return wrapper
 9 #
10 # @show_time# foo=show_time(foo)
11 # def foo(n):
12 #     if n==1:
13 #         return 1
14 #     return n*foo(n-1)
15 # print(foo(6))

这种现象是不对的,foo变量已经指向了wrapper函数在内存中的地址了。要解决这种问题必须在wrapper里面接收原func返回的结果,并把这个结果返回回去,代码如下所示。还有一种方式就是用另一个普通函数将递归函数包裹起来,然后使用普通装饰器去装饰该普通函数。

 1 def show_time(func):
 2
 3     def wrapper(n):
 4         ret=func(n)
 5         print("hello,world")
 6         return ret
 7     return wrapper
 8
 9 @show_time# foo=show_time(foo)
10 def foo(n):
11     def _foo(n):
12         if n==1:
13             return 1
14         return n*_foo(n-1)
15     return _foo(n)
16 print(foo(6))

时间: 2025-01-12 21:31:34

python高级之装饰器的相关文章

Python深入05 装饰器

作者:Vamei 出处:http://www.cnblogs.com/vamei 欢迎转载,也请保留这段声明.谢谢! 装饰器(decorator)是一种高级Python语法.装饰器可以对一个函数.方法或者类进行加工.在Python中,我们有多种方法对函数和类进行加工,比如在Python闭包中,我们见到函数对象作为某一个函数的返回结果.相对于其它方式,装饰器语法简单,代码可读性高.因此,装饰器在Python项目中有广泛的应用. 装饰器最早在Python 2.5中出现,它最初被用于加工函数和方法这样

python学习笔记--装饰器

1.首先是一个很无聊的函数,实现了两个数的加法运算: def f(x,y): print x+y f(2,3) 输出结果也ok 5 2.可是这时候我们感觉输出结果太单一了点,想让代码的输出多一点看起来不这么单调: def showInfo(fun): def wrap(x,y): print "The function before" func(x,y) print "The function after" return wrap def f(x,y): print

1.16 Python基础知识 - 装饰器

Python中的装饰器就是函数,作用就是包装其他函数,为他们起到修饰作用.在不修改源代码的情况下,为这些函数额外添加一些功能,像日志记录,性能测试等.一个函数可以使用多个装饰器,产生的结果与装饰器的位置顺序有关. 装饰器基本形式: @装饰器1 def 函数1: 函数体 相当于:==> 函数1 = 装饰器1(函数1) 装饰器特点: 1.不修改源代码的调用方式 2.不修改源代码内容 3.装饰器有高阶函数与递归函数相融合的特点 多个装饰器修饰,示例: @foo @spam def bar():pass

ZMAN的学习笔记之Python篇:装饰器

年前工作事务比较繁琐,我只能用零碎的时间继续学习Python,决定开一个系列的博文,作为自己深入学习Python的记录吧.名字也取好了,就叫<ZMAN的学习笔记之Python篇>~开篇是关于装饰器的,春节假期码的字哈哈~就让我们开始吧! 本文的例子都是自己想的,如果不是很合适,请大家提出宝贵意见哈~谢谢啦! 一.为什么要用“装饰器” 比如我们写了如下一段代码: # 打印0~99 def func(): for i in range(100): print(i) 我们想要监测执行这个函数花费了多

六、PYTHON 学习之装饰器使用

Python是一种强大的语言,即可浅尝辄止,也可深入挖掘.很适合做科学计算.数据挖掘等等.今天我将简单介绍一下Python的装饰器(Decorators)的用法 . 假设我们想要庆祝下生日,需要邀请一些朋友过来参加.但是你有个讨厌的朋友,叫Joe,必须不能让他来啊.可能首先你想到的是建一个list,然后迭代查找并移除所有的Joe童鞋.这当然是个好方法,但是这里为了介绍装饰器,我们会用@来完成这个工作.虽然可能看起来没有什么必要,但是有助于大家学习装饰器的用法. 首先创建一个Python文件app

python笔记 - day4-之装饰器

             python笔记 - day4-之装饰器 需求: 给f1~f100增加个log: def outer(): #定义增加的log print("log") def f1(): outer() #分别调用函数 print("F1") def f2(): outer() #分别调用函数 print("F2") def f100(): outer() #分别调用函数 print("F100") f1() f2

[python基础]关于装饰器

在面试的时候,被问到装饰器,在用的最多的时候就@classmethod ,@staticmethod,开口胡乱回答想这和C#的static public 关键字是不是一样的,等面试回来一看,哇,原来是这样,真佩服我当时厚着脸皮回答的那些问题... OK,先来张图看看装饰器内容: OK,我们留下一个印象,然后我们看实际的场景来操作. 我们先看一个方法: __author__ = 'bruce' def do_sth(): print 'some thing has been done' if __

@修饰符--python中的装饰器

http://blog.csdn.net/shangzhihaohao/article/details/6928808 装饰器模式可以在不影响其他对象的情况下,以动态.透明的方式给单个对象添加职责,也能够处理那些可以撤销的职责.经常用于日志记录.性能测试等场合. 想象一下这个很常见的场景,你写了一个方法只提供给以登陆的用户访问(事实上我也是通过django的@login_required才了解到@修饰符的),你可以写以下代码: 这当然没什么问题,但是你又写了一个方法B,也要求只有登录用户可以访问

Python 中实现装饰器时使用 @functools.wraps 的理由

Python 中使用装饰器对在运行期对函数进行一些外部功能的扩展.但是在使用过程中,由于装饰器的加入导致解释器认为函数本身发生了改变,在某些情况下--比如测试时--会导致一些问题.Python 通过 functool.wraps 为我们解决了这个问题:在编写装饰器时,在实现前加入 @functools.wraps(func) 可以保证装饰器不会对被装饰函数造成影响.比如,在 Flask 中,我们要自己重写 login_required 装饰器,但不想影响被装饰器装饰的方法,则 login_req