Python的 decorator 本质上就是一个高阶函数,它接收一个函数作为参数,然后,返回一个新函数。

使用 decorator 用Python提供的 @ 语法,这样可以避免手动编写 f = decorate(f) 这样的代码。

考察一个@log的定义:

def log(f):
def fn(x):
print 'call ' + f.__name__ + '()...'
return f(x)
return fn

对于阶乘函数,@log工作得很好:

@log
def factorial(n):
return reduce(lambda x,y: x*y, range(1, n+1))
print factorial(10)

结果:

call factorial()...
3628800

但是,对于参数不是一个的函数,调用将报错:

@log
def add(x, y):
return x + y
print add(1, 2)

结果:

Traceback (most recent call last):
File "test.py", line 15, in <module>
print add(1,2)
TypeError: fn() takes exactly 1 argument (2 given)

因为 add() 函数需要传入两个参数,但是 @log 写死了只含一个参数的返回函数。

要让 @log 自适应任何参数定义的函数,可以利用Python的 *args 和 **kw,保证任意个数的参数总是能正常调用:

def log(f):
def fn(*args, **kw):
print 'call ' + f.__name__ + '()...'
return f(*args, **kw)
return fn

现在,对于任意函数,@log 都能正常工作。

练习:

编写一个@performance,它可以打印出函数调用的时间

import time
def performance(f):
a = time.time()
def fn(*argc, **kw):
print f(*argc, **kw)
print time.time() - a
return fn
@performance
def factorial(n):
return reduce(lambda x,y: x*y, range(1, n+1))
print factorial(10)

结果:

3628800
0.0189998149872
None

最新文章

  1. React Native 之 TextInput使用
  2. JSF primefaces session view expired 会话失效后页面跳转
  3. JVM--标记-清除算法Mark-Sweep
  4. python property
  5. jQuery邮箱自动补全代码
  6. Hibernate的配置文件以及用法
  7. 小白科普之JavaScript的数组
  8. cookie函数
  9. C#:数据交互
  10. 15条JavaScript最佳实践很经典噢
  11. Oracle 动态视图4 V$SESSION_WAIT &amp; V$SESSION_EVENT
  12. JQuery重要知识点
  13. Vim配置说明
  14. openui5中的RESTful实现odata协议
  15. NHibernate的基本使用
  16. Go语言学习笔记(五) [函数]
  17. YUV420、YUV422、RGB24转换
  18. java反射修改final变量
  19. 解决ScrollView嵌套RecyclerView出现item显示不全的问题
  20. 关于win10下JDK环境变量的配置以及关于JDK的一些说明

热门文章

  1. diamond源码阅读-diamond-client
  2. Android-ViewPagerIndicator框架使用——TabPageIndicator
  3. SpringAOP和AspectJ
  4. Jmeter与LoadRunner 测试Java项目的坑
  5. 免安装mysql配置
  6. IOS7 UI Transition Guide 状态栏 statusbar
  7. Android开发:《Gradle Recipes for Android》阅读笔记1.7——仓库配置
  8. 使用Nexus管理Maven仓库时,上传带依赖的第三方jar
  9. ehcache缓存框架简介(一)
  10. ffmpeg参数使用说明2