Python 装饰器(Decorator)
装饰器的语法为 @dec_name ,置于函数定义之前。如:
import atexit @atexit.register
def goodbye():
print('Goodbye!') print('Script end here')
atexit.register 是一个装饰器,它的作用是将被装饰的函数注册为在程序结束时执行。函数 goodbye 是被装饰的函数。
程序的运行结果是:
Script end here
Goodbye!
可见函数 goodbye 在程序结束后被自动调用。
另一个常见的例子是 @property ,装饰类的成员函数,将其转换为一个描述符。
class Foo:
@property
def attr(self):
print('attr called')
return 'attr value' foo = Foo()
等价语法
语句块
@atexit.register
def goodbye():
print('Goodbye!')
等价于
def goodbye():
print('Goodbye!')
goodbye = atexit.register(goodbye)
这两种写法在作用上完全等价。
从第二种写法,更容易看出装饰器的原理。装饰器实际上是一个函数(或callable),其输入、返回值为:
| 说明 | 示例中的对应 | |
|---|---|---|
| 输入 | 被装饰的函数 | goodbye |
| 返回值 | 变换后的函数或任意对象 |
返回值会被赋值给原来指向输入函数的变量,如示例中的 goodbye 。此时变量 goodbye 将指向装饰器的返回值,而不是原来的函数定义。返回值一般为一个函数,这个函数是在输入参数函数添加了一些额外操作的版本。
如下面这个装饰器对原始函数添加了一个操作:每次调用这个函数时,打印函数的输入参数及返回值。
def trace(func):
def wrapper(*args, **kwargs): 1
print('Enter. Args: %s, kwargs: %s' % (args, kwargs)) 2
rv = func(*args, **kwargs) 3
print('Exit. Return value: %s' % (rv)) 4
return rv return wrapper @trace
def area(height, width):
print('area called')
return height * width area(2, 3) 5
- 1 :定义一个新函数,这个函数将作为装饰器的返回值,来替换原函数
- 2, 4 : 打印输入参数、返回值。这是这个装饰器所定义的操作
- 3 :调用原函数
- 5 :此时
area实际上是 1 处定义的wrapper函数
程序的运行结果为:
Enter. Args: (2, 3), kwargs: {}
area called
Exit. Return value: 6
如果不使用装饰器,则必须将以上打印输入参数及返回值的语句直接写在 area 函数里,如:
def area(height, width):
print('Enter. Args: %s, %s' % (height, width))
print('area called')
rv = height * width
print('Exit. Return value: %s' % (rv))
return rv area(2, 3)
程序的运行结果与使用装饰器时相同。但使用装饰器的好处为:
- 打印输入参数及返回值这个操作可被重用
如对于一个新的函数
foo,装饰器trace可以直接拿来使用,而无须在函数内部重复写两条print语句。@trace
def foo(val):
return 'return value'一个装饰器实际上定义了一种可重复使用的操作
- 函数的功能更单纯
area函数的功能是计算面积,而调试语句与其功能无关。使用装饰器可以将与函数功能无关的语句提取出来。 因此函数可以写地更小。使用装饰器,相当于将两个小函数组合起来,组成功能更强大的函数
修正名称
以上例子中有一个缺陷,函数 area 被 trace 装饰后,其名称变为 wrapper ,而非 area 。 print(area) 的结果为:
<function wrapper at 0x10df45668>
wrapper 这个名称来源于 trace 中定义的 wrapper 函数。
可以通过 functools.wraps 来修正这个问题。
from functools import wraps def trace(func):
@wraps(func)
def wrapper(*args, **kwargs):
print('Enter. Args: %s, kwargs: %s' % (args, kwargs))
rv = func(*args, **kwargs)
print('Exit. Return value: %s' % (rv))
return rv return wrapper @trace
def area(height, width):
print('area called')
return height * width
即使用 functools.wraps 来装饰 wrapper 。此时 print(area) 的结果为:
<function area at 0x10e8371b8>
函数的名称能够正确显示。
接收参数
以上例子中 trace 这个装饰器在使用时不接受参数。如果想传入参数,如传入被装饰函数的名称,可以这么做:
from functools import wraps def trace(name):
def wrapper(func):
@wraps(func)
def wrapped(*args, **kwargs):
print('Enter %s. Args: %s, kwargs: %s' % (name, args, kwargs))
rv = func(*args, **kwargs)
print('Exit %s. Return value: %s' % (name, rv))
return rv return wrapped
return wrapper @trace('area')
def area(height, width):
print('area called')
return height * width area(2, 3)
程序的运行结果为:
Enter area. Args: (2, 3), kwargs: {}
area called
Exit area. Return value: 6
将函数名称传入后,在日志同时打印出函数名,日志更加清晰。
@trace('area') 是如何工作的?
这里其实包含了两个步骤。 @trace('area') 等价于:
dec = trace('area')
@dec
def area(height, width): ...
即先触发函数调用 trace('area') ,得到一个返回值,这个返回值为 wrapper 函数。 而这个函数才是真正的装饰器,然后使用这个装饰器装饰函数。
多重装饰器
装饰器可以叠加使用,如:
@dec1
@dec2
def foo():pass
等价的代码为:
def foo():pass
foo = dec2(foo)
foo = dec1(foo)
即装饰器依次装饰函数,靠近函数定义的装饰器优先。相当于串联起来。
如果你一路读到这里,我相信你已经掌握了关于Python 装饰器80%的知识,并能够应用到工作学习中。你知道了装饰器的工作原理,以及如何自己编写一个装饰器,及避免常见的编写错误。
如果你仍然有疑问,欢迎留言讨论!(如果你觉得这篇文章有用,请点下方推荐按钮:p)
Python 装饰器(Decorator)的更多相关文章
- python 装饰器(decorator)
装饰器(decorator) 作者:Vamei 出处:http://www.cnblogs.com/vamei 欢迎转载,也请保留这段声明.谢谢! 装饰器(decorator)是一种高级Python语 ...
- Python装饰器--decorator
装饰器 装饰器实质是一个函数,其作用就是在不改动其它函数代码的情况下,增加一些功能.如果我们需要打印函数调用前后日志,可以这么做 def log(func): print('%s is running ...
- Python 装饰器Decorator(一)
(一) 装饰器基础知识 什么是Python装饰器?Python里装饰器是一个可调用的对象(函数),其参数是另一个函数(被装饰的函数) 假如有一个名字为somedecorator的装饰器,target是 ...
- Python装饰器(Decorator)简介
Python有许多出色的语言特性,装饰器(Decorator)便是其中一朵奇葩.先来看看一段代码: def deco1(f): print 'decorate 1' return f def deco ...
- Python 装饰器Decorator(二)
对于上一篇“”Python闭包“”随笔中提到的make_averager()函数的如下实现,我们把历史值保存在列表里,每次计算平均值都需要重新求和,当历史值较多时,需要占用比较多的空间并且效率也不高. ...
- python 装饰器学习(decorator)
最近看到有个装饰器的例子,没看懂, #!/usr/bin/python class decorator(object): def __init__(self,f): print "initi ...
- python语法32[装饰器decorator](转)
一 装饰器decorator decorator设计模式允许动态地对现有的对象或函数包装以至于修改现有的职责和行为,简单地讲用来动态地扩展现有的功能.其实也就是其他语言中的AOP的概念,将对象或函数的 ...
- python 语法之 装饰器decorator
装饰器 decorator 或者称为包装器,是对函数的一种包装. 它能使函数的功能得到扩充,而同时不用修改函数本身的代码. 它能够增加函数执行前.执行后的行为,而不需对调用函数的代码做任何改变. 下面 ...
- python函数编程-装饰器decorator
函数是个对象,并且可以赋值给一个变量,通过变量也能调用该函数: >>> def now(): ... print('2017-12-28') ... >>> l = ...
- Python——装饰器(Decorator)
1.什么是装饰器? 装饰器放在一个函数开始定义的地方,它就像一顶帽子一样戴在这个函数的头上.和这个函数绑定在一起.在我们调用这个函数的时候,第一件事并不是执行这个函数,而是将这个函数做为参数传入它头顶 ...
随机推荐
- Guava 教程(3):Java 的函数式编程,通过 Google Collections 过滤和调用
原文出处: oschina 在本系列博客的第一.二部分,我介绍了非常优秀的Google Collections和Guava包.本篇博客中我们来看看如何使用Google Collections来做到过滤 ...
- Python基础 语法特别注意笔记(和Java相比)
Python变量和数据类型 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 ...
- XBMC源代码简析 5:视频播放器(dvdplayer)-解复用器(以ffmpeg为例)
XBMC分析系列文章: XBMC源代码分析 1:整体结构以及编译方法 XBMC源代码分析 2:Addons(皮肤Skin) XBMC源代码分析 3:核心部分(core)-综述 XBMC源代码分析 4: ...
- (五十九)iOS网络基础之UIWebView简易浏览器实现
[UIWebView网络浏览器] 通过webView的loadRequest方法可以发送请求显示相应的网站,例如: NSURL *url = [NSURL URLWithString:@"h ...
- 【算法导论】最小生成树之Prime法
关于最小生成树的概念,在前一篇文章中已经讲到,就不在赘述了.下面介绍Prime算法: 其基本思想为:从一个顶点出发,选择由该顶点出发的最小权值边,并将该边的另一个顶点包含进来,然后找出 ...
- UE4 创建第三人称角色
一.C++创建 1. 创建一个C++类,继承自Character,取名MyThirdCharacter 2. 在头文件中声明一个摄像机目标点CameraBoom,一个摄像机Foll ...
- Java进阶(十四)实现每天定时对数据库的操作
Java实现每天定时对数据库操作 现在有一个很棘手的问题:客户要求实现一个功能,就是每日凌晨自动计算慢性病订单是否有需要在今日提醒的,如果有则生成一条提醒记录到lm_notice之中. 如何在Web工 ...
- 【一天一道LeetCode】#9. Palindrome Number
一天一道LeetCode系列 (一)题目 Determine whether an integer is a palindrome. Do this without extra space. Some ...
- C++模板总结
在编写含有模板的程序的时候,我还是按照一个头文件声明,一个源文件的方法来组织,结果编译的时候总出现一些很奇怪的语法问题,但程序明明是没有问题的.后来经过查阅才知道原来是因为C++编译器不支持对模板的分 ...
- 【编程练习】快速select算法的实现
代码来自: http://blog.csdn.net/v_JULY_v 算法思想: // Quick_select.cpp : 定义控制台应用程序的入口点. // #include "std ...