python中完善decorator
2024-08-30 13:29:02
@decorator可以动态实现函数功能的增加,但是,经过@decorator“改造”后的函数,和原函数相比,除了功能多一点外,有没有其它不同的地方?
在没有decorator的情况下,打印函数名:
def f1(x):
pass
print f1.__name__
输出:
f1
有decorator的情况下,再打印函数名:
def log(f):
def wrapper(*args, **kw):
print 'call...'
return f(*args, **kw)
return wrapper
@log
def f2(x):
pass
print f2.__name__
输出:
wrapper
可见,由于decorator返回的新函数函数名已经不是'f2',而是@log内部定义的'wrapper'。这对于那些依赖函数名的代码就会失效。decorator还改变了函数的__doc__等其它属性。如果要让调用者看不出一个函数经过了@decorator的“改造”,就需要把原函数的一些属性复制到新函数中:
def log(f):
def wrapper(*args, **kw):
print 'call...'
return f(*args, **kw)
wrapper.__name__ = f.__name__
wrapper.__doc__ = f.__doc__
return wrapper
这样写decorator很不方便,因为我们也很难把原函数的所有必要属性都一个一个复制到新函数上,所以Python内置的functools可以用来自动化完成这个“复制”的任务:
import functools
def log(f):
@functools.wraps(f)
def wrapper(*args, **kw):
print 'call...'
return f(*args, **kw)
return wrapper
最后需要指出,由于我们把原函数签名改成了(*args, **kw),因此,无法获得原函数的原始参数信息。即便我们采用固定参数来装饰只有一个参数的函数:
def log(f):
@functools.wraps(f)
def wrapper(x):
print 'call...'
return f(x)
return wrapper
也可能改变原函数的参数名,因为新函数的参数名始终是 'x',原函数定义的参数名不一定叫 'x'。
最新文章
- install scrapy-redis on centos
- [SharePoint 2010] Modify lookup mapping with PowerShell
- php怎么解决超链接中的中文参数转码问题?
- 华为HG255D路由器使用OH3C进行中大校园网认证
- 利用CSS的@font-face属性 在网页中嵌入字体
- 最初程序员的思维“修炼”之四——Android平台开发的“强制关闭”解决思路
- director.js教程
- Collecting Bugs poj2096 概率DP
- Ubuntu下搜狗拼音突然无法输入中文的解决办法
- C#中Abstract和Virtual[转载]
- linux下 ls -l 命令显示结果每一列代表什么意思
- Java集合-----java集合框架常见问题
- python绘制等边三角形
- 你电梯没了—OO第二单元作业思考
- Redis AOF、RDB持久化
- GCD XOR(UVa 12716)
- listagg乱码问题
- 【Linux】Linux中Swap与Memory内存简单介绍
- diskpart 格式化u盘 制作u盘启动盘方法
- [翻译] 扩张卷积 (Dilated Convolution)