转自:http://guangboo.org/2013/01/28/python-mixin-programming


Mixin 简介

Mixin 编程是一种开发模式,是一种将多个类中的功能单元的进行组合的利用的方式,这听起来就像是有类的继承机制就可以实现,然而这与传统的类继承有所不同。通常 Mixin 并不作为任何类的基类,也不关心与什么类一起使用,而是在运行时动态的同其他零散的类一起组合使用。


特点

使用 Mixin 机制有如下好处:

  • 可以在不修改任何源代码的情况下,对已有类进行扩展;
  • 可以保证组件的划分;
  • 可以根据需要,使用已有的功能进行组合,来实现“新”类;很
  • 好的避免了类继承的局限性,因为新的业务需要可能就需要创建新的子类。



多继承

Python支持多继承,即一个类可以继承多个子类。可以利用该特性,可以方便的实现mixin继承。如下代码,类A,B分别表示不同的功能单元,C为A,B功能的组合,这样类C就拥有了类A, B的功能。

  1. class A:
  2.     def get_a(self):
  3.     print 'a'
  4. class B:
  5.     def get_b(self):
  6.     print 'b'
  7. class C(A, B): 
  8.     pass
  9. c = C()
  10. c.get_a()
  11. c.get_b()
复制代码



__bases__

多继承的实现就会创建新类,有时,我们在运行时,希望给类A添加类B的功能时,也可以利用python的元编程特性,__bases__属性便在运行时轻松给类A添加类B的特性,如下代码:

  1. A.__bases__ += (B,)
  2. a.get_b()
复制代码


其实__bases__也是继承的机制,因为__bases__属性存储了类的基类。因此多继承的方法也可以这样实现:

  1. class C:
  2.     pass
  3. C.__bases__ += (A, B, )
复制代码



插件方式

以上两种方式,都是基于多继承和python的元编程特性,然而在业务需求变化时,就需要新的功能组合,那么就需要重新修改A的基类,这回带来同步的问题,因为我们改的是类的特性,而不是对象的。因此以上修改会对所有引用该类的模块都收到影响,这是相当危险的。通常我们希望修改对象的行为,而不是修改类的。同样的我们可以利用__dict__来扩展对象的方法。

  1. class PlugIn(object):
  2.     def __init__(self):
  3.         self._exported_methods = []
  4.         
  5.     def plugin(self, owner):
  6.         for f in self._exported_methods:
  7.             owner.__dict__[f.__name__] = f
  8.     def plugout(self, owner):
  9.         for f in self._exported_methods:
  10.             del owner.__dict__[f.__name__]
  11. class AFeature(PlugIn):
  12.     def __init__(self):
  13.         super(AFeature, self).__init__()
  14.         self._exported_methods.append(self.get_a_value)
  15.     def get_a_value(self):
  16.         print 'a feature.'
  17. class BFeature(PlugIn):
  18.     def __init__(self):
  19.         super(BFeature, self).__init__()
  20.         self._exported_methods.append(self.get_b_value)
  21.     def get_b_value(self):
  22.         print 'b feature.'
  23. class Combine:pass
  24. c = Combine()
  25. AFeature().plugin(c)
  26. BFeature().plugin(c)
  27. c.get_a_value()
  28. c.get_b_value()
复制代码
 posted on 2014-11-14 21:18  大雄fcl  阅读(309)  评论(0编辑  收藏  举报