利用metaclass实现python的aop
修行Python个把星期,终于有点成果了,呵呵,一个利用metaclass实现的aop。
其实python这类非常动态的语言要实现AOP是很容易的,所以首先我们要来先定义一个metaclass
然后我们要在__new__()这个metaclass 的时候动态植入方法到要调用地方法的前后。
具体代码如下:

2
__author__="alex"3
__date__ ="$2008-12-5 23:54:11$"4
__name__="pyaop"5

6
'''7
这个metaclass是实现AOP的基础8
'''9
class pyaop(type):10
'''11
这个空方法是用来将后面的beforeop和afterop初始化成函数引用12
'''13
def nop(self):14
pass15
'''16
下面这两个变量是类变量,也就是存放我们要植入的两个函数的地址的变量17
'''18
beforeop=nop19
afterop=nop20
'''21
设置前后两个植入函数的类函数22
'''23
@classmethod24
def setbefore(self,func):25
pyaop.beforeop=func26
@classmethod27
def setafter(self,func):28
pyaop.afterop=func29
'''30
初始化metaclass的函数,这个函数最重要的就是第四个参数,dict通过这个参数我们可以修改类的属性(方法)31
'''32
def __new__(mcl,name,bases,dict):33
from types import FunctionType #加载类型模块的FunctionType34
obj=object() #定义一个空对象的变量35
'''36
这个就是要植入的方法,func参数就是我们要调用的函数37
'''38
def aop(func):39
'''40
我们用这个函数来代替将要调用的函数41
'''42
def wrapper(*args, **kwds):43
pyaop.beforeop(obj) #调用前置函数44
value = func(*args, **kwds) #调用本来要调用的函数45
pyaop.afterop(obj) #调用后置函数46
return value #返回47
return wrapper48
#在类的成员列表中查找即将调用的函数49
for attr, value in dict.iteritems():50
if isinstance(value, FunctionType):51
dict[attr] = aop(value) #找到后用aop这个函数替换之52
obj=super(pyaop, mcl).__new__(mcl, name, bases, dict) #调用父类的__new__()创建self53
return obj54
使用的时候,如果我们要拦截一个类A的方法调用,就这样子:
class A(object):2
__metaclass__ = pyaop3
def foo(self):4
total = 05
for i in range(100000):6
total = total+17
print total8

9
def foo2(self):10
from time import sleep11
total = 012
for i in range(100000):13
total = total+114
sleep(0.0001)15
print total
最后我们只需要:
def beforep(self):2
print('before')3
def afterp(self):4
print('after')5

6
if __name__ == "__main__":7
pyaop.setbefore(beforep)8
pyaop.setafter(afterp)9
a=A()10
a.foo()11
a.foo2()这样子在执行代码的时候就得到了如下结果
before
100000
after
before
100000
after
这段代码耗时一天才调通(汗颜),是基于园里另外一贴 http://www.cnblogs.com/cavingdeep/archive/2006/08/22/483056.html
并感谢python讨论群69828975里的大虾Seewind(310380)给予的无私帮助
欢迎各位python大牛给予斧正
浙公网安备 33010602011771号