设为首页 收藏本站
查看: 1193|回复: 0

[经验分享] 利用metaclass实现python的aop

[复制链接]

尚未签到

发表于 2015-4-25 10:38:41 | 显示全部楼层 |阅读模式
  修行Python个把星期,终于有点成果了,呵呵,一个利用metaclass实现的aop。
  其实python这类非常动态的语言要实现AOP是很容易的,所以首先我们要来先定义一个metaclass
  然后我们要在__new__()这个metaclass 的时候动态植入方法到要调用地方法的前后。

  具体代码如下:

1 DSC0000.gif
2__author__="alex"
3__date__ ="$2008-12-5 23:54:11$"
4__name__="pyaop"
5
6'''
7这个metaclass是实现AOP的基础
8'''
9class pyaop(type):
10    '''
11    这个空方法是用来将后面的beforeop和afterop初始化成函数引用
12    '''
13    def nop(self):
14        pass
15    '''
16    下面这两个变量是类变量,也就是存放我们要植入的两个函数的地址的变量
17    '''
18    beforeop=nop
19    afterop=nop
20    '''
21    设置前后两个植入函数的类函数
22    '''
23    @classmethod
24    def setbefore(self,func):
25        pyaop.beforeop=func
26    @classmethod
27    def setafter(self,func):
28        pyaop.afterop=func
29    '''
30    初始化metaclass的函数,这个函数最重要的就是第四个参数,dict通过这个参数我们可以修改类的属性(方法)
31    '''
32    def __new__(mcl,name,bases,dict):
33        from types import FunctionType #加载类型模块的FunctionType
34        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 wrapper
48        #在类的成员列表中查找即将调用的函数
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__()创建self
53        return obj
54      
  使用的时候,如果我们要拦截一个类A的方法调用,就这样子:
  

1class A(object):
2    __metaclass__ = pyaop
3    def foo(self):
4        total = 0
5        for i in range(100000):
6            total = total+1
7        print total
8
9    def foo2(self):
10        from time import sleep
11        total = 0
12        for i in range(100000):
13            total = total+1
14            sleep(0.0001)
15        print total  
  最后我们只需要:
  

1def beforep(self):
2    print('before')
3def afterp(self):
4    print('after')
5
6if __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.iyunv.com/cavingdeep/archive/2006/08/22/483056.html
  并感谢python讨论群69828975里的大虾Seewind(310380)给予的无私帮助
  欢迎各位python大牛给予斧正

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-60501-1-1.html 上篇帖子: Windows上面安装MySQL-Python-1.2.3(Mysqldb- error: Unable to find vcvarsall.bat) 下篇帖子: GDAL(ZZ) Python
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表