写在前面的
不管是过去还是现在,二总是个贬义词。可是,我却喜欢这个数字。这证明,在你做出初步尝试后,成功的往前踏了一步。
装饰器的类比
装饰器在C#中可以类比Attribute,实现在方法执行前或执行后进行初始化或者扫尾工作。装饰器的实现实际上基于函数式编程。即传递一个函数参数,进行某些操作后,返回装饰加工之后的函数。这个返回的函数也许是原函数的偏函数,也许是另外一个函数。
既然是基于私有函数,那么可能就会存在父函数的作用域。由此带来了编程中的流行概念闭包。
装饰器解释
装饰器最早在Python 2.5中出现,它最初被用于加工函数和方法这样的可调用对象(callable object,这样的对象定义有__call__方法)。在Python 2.6以及之后的Python版本中,装饰器被进一步用于加工类。(引用于http://www.cnblogs.com/vamei/archive/2013/02/16/2820212.html)
从本质上讲,这些特征引入了C#开发者称之为AOP(面向方面编程)的概念。你可以考虑在装饰器中置入通用功能的代码来降低程序复杂度。例如,可以用装饰器来实现:
1、 引入日志
2、 给函数加入事务能力
装饰器举例
def decorator(F):
def new_F(a, b):
print("input", a, b)
return F(a, b)
return new_F
# get square sum
@decorator
def square_sum(a, b):
return a**2 + b**2
# get square diff
@decorator
def square_diff(a, b):
return a**2 - b**2
print(square_sum(3, 4))
print(square_diff(3, 4))
从上面代码中可以发现,装饰器函数一定会返回一个加工后的函数,否则,调用被装饰函数时会弹出空引用。
闭包
闭包是基于父函数名字空间被外部引用,《python核心编程》一书作者称其为流浪作用域。但是这些流浪者会带来有意思的故事。
闭包与装饰器联合举例
#!/usr/bin/env python
from time import time
def logged(when):
def log(f,*args,**kargs):
print '''called:
function:%s
args:%r
kargs:%r''' % (f,args,kargs)
def pre_logged(f):
def wrapper(*args,**kargs):
log(f,*args,**kargs)
return f(*args,**kargs)
return wrapper
def post_logged(f):
def wrapper(*args,**kargs):
now=time()
try:
return f(*args,**kargs)
finally:
log(f,*args,**kargs)
print "time delta:%s" % (time()-now)
return wrapper
try:
return {'pre':pre_logged,'post':post_logged}[when]
except KeyError,e:
raise ValueError(e),'must be "pre" or "post"'
@logged('pre')
def hello(name):
print 'hello,',name
hello('world')
这段例子中有三层函数,请一定要注意函数的作用域。如果理解了函数作用域后,该段代码比较好理解。
由于是首次学习Python,所以博客水平有限,望大家多多支持。