修行Python个把星期,终于有点成果了,呵呵,一个利用metaclass实现的aop。

其实python这类非常动态的语言要实现AOP是很容易的,所以首先我们要来先定义一个metaclass

然后我们要在__new__()这个metaclass 的时候动态植入方法到要调用地方法的前后。

具体代码如下:

 1利用metaclass实现python的aop
 2利用metaclass实现python的aop__author__="alex"
 3利用metaclass实现python的aop__date__ ="$2008-12-5 23:54:11$"
 4利用metaclass实现python的aop__name__="pyaop"
 5利用metaclass实现python的aop
 6利用metaclass实现python的aop'''
 7利用metaclass实现python的aop这个metaclass是实现AOP的基础
 8利用metaclass实现python的aop'''
 9利用metaclass实现python的aopclass pyaop(type):
10利用metaclass实现python的aop    '''
11利用metaclass实现python的aop    这个空方法是用来将后面的beforeop和afterop初始化成函数引用
12利用metaclass实现python的aop    '''
13利用metaclass实现python的aop    def nop(self):
14利用metaclass实现python的aop        pass
15利用metaclass实现python的aop    '''
16利用metaclass实现python的aop    下面这两个变量是类变量,也就是存放我们要植入的两个函数的地址的变量
17利用metaclass实现python的aop    '''
18利用metaclass实现python的aop    beforeop=nop
19利用metaclass实现python的aop    afterop=nop
20利用metaclass实现python的aop    '''
21利用metaclass实现python的aop    设置前后两个植入函数的类函数
22利用metaclass实现python的aop    '''
23利用metaclass实现python的aop    @classmethod
24利用metaclass实现python的aop    def setbefore(self,func):
25利用metaclass实现python的aop        pyaop.beforeop=func
26利用metaclass实现python的aop    @classmethod
27利用metaclass实现python的aop    def setafter(self,func):
28利用metaclass实现python的aop        pyaop.afterop=func
29利用metaclass实现python的aop    '''
30利用metaclass实现python的aop    初始化metaclass的函数,这个函数最重要的就是第四个参数,dict通过这个参数我们可以修改类的属性(方法)
31利用metaclass实现python的aop    '''
32利用metaclass实现python的aop    def __new__(mcl,name,bases,dict):
33利用metaclass实现python的aop        from types import FunctionType #加载类型模块的FunctionType
34利用metaclass实现python的aop        obj=object() #定义一个空对象的变量
35利用metaclass实现python的aop        '''
36利用metaclass实现python的aop        这个就是要植入的方法,func参数就是我们要调用的函数
37利用metaclass实现python的aop        '''
38利用metaclass实现python的aop        def aop(func):
39利用metaclass实现python的aop            '''
40利用metaclass实现python的aop            我们用这个函数来代替将要调用的函数
41利用metaclass实现python的aop            '''
42利用metaclass实现python的aop            def wrapper(*args, **kwds):
43利用metaclass实现python的aop                pyaop.beforeop(obj) #调用前置函数
44利用metaclass实现python的aop                value = func(*args, **kwds) #调用本来要调用的函数
45利用metaclass实现python的aop                pyaop.afterop(obj) #调用后置函数
46利用metaclass实现python的aop                return value #返回
47利用metaclass实现python的aop            return wrapper
48利用metaclass实现python的aop        #在类的成员列表中查找即将调用的函数
49利用metaclass实现python的aop        for attr, value in dict.iteritems():
50利用metaclass实现python的aop            if isinstance(value, FunctionType):
51利用metaclass实现python的aop                dict[attr] = aop(value) #找到后用aop这个函数替换之
52利用metaclass实现python的aop        obj=super(pyaop, mcl).__new__(mcl, name, bases, dict) #调用父类的__new__()创建self
53利用metaclass实现python的aop        return obj
54利用metaclass实现python的aop    

 

使用的时候,如果我们要拦截一个类A的方法调用,就这样子:

 

 1利用metaclass实现python的aopclass A(object):
 2利用metaclass实现python的aop    __metaclass__ = pyaop
 3利用metaclass实现python的aop    def foo(self):
 4利用metaclass实现python的aop        total = 0
 5利用metaclass实现python的aop        for i in range(100000):
 6利用metaclass实现python的aop            total = total+1
 7利用metaclass实现python的aop        print total
 8利用metaclass实现python的aop
 9利用metaclass实现python的aop    def foo2(self):
10利用metaclass实现python的aop        from time import sleep
11利用metaclass实现python的aop        total = 0
12利用metaclass实现python的aop        for i in range(100000):
13利用metaclass实现python的aop            total = total+1
14利用metaclass实现python的aop            sleep(0.0001)
15利用metaclass实现python的aop        print total

 

最后我们只需要:

 

 1利用metaclass实现python的aopdef beforep(self):
 2利用metaclass实现python的aop    print('before')
 3利用metaclass实现python的aopdef afterp(self):
 4利用metaclass实现python的aop    print('after')
 5利用metaclass实现python的aop
 6利用metaclass实现python的aopif __name__ == "__main__":
 7利用metaclass实现python的aop    pyaop.setbefore(beforep)
 8利用metaclass实现python的aop    pyaop.setafter(afterp)
 9利用metaclass实现python的aop    a=A()
10利用metaclass实现python的aop    a.foo()
11利用metaclass实现python的aop    a.foo2()

 这样子在执行代码的时候就得到了如下结果

before
100000
after
before
100000
after

这段代码耗时一天才调通(汗颜),是基于园里另外一贴 http://www.cnblogs.com/cavingdeep/archive/2006/08/22/483056.html 

并感谢python讨论群69828975里的大虾Seewind(310380)给予的无私帮助

欢迎各位python大牛给予斧正

相关文章: