Mixin模式是一種在python里經(jīng)常使用的模式裸卫,適當(dāng)合理的應(yīng)用能夠達(dá)到復(fù)用代碼,合理組織代碼結(jié)構(gòu)的目的企软。
Python的Mixin模式可以通過多繼承
的方式來實(shí)現(xiàn), 舉例來說危彩,我們自定義一個(gè)簡單的具有嵌套結(jié)構(gòu)的數(shù)據(jù)容器:
class SimpleItemContainer(object):
def __init__(self, id, item_containers):
self.id = id
self.data = {}
for item in item_containers:
self.data[item.id] = item
SimpleItemContainer
通過python內(nèi)置類型Dict
來存放數(shù)據(jù)炼杖,不過到目前為止想要訪問對應(yīng)的數(shù)據(jù)還是得直接調(diào)用里面的字典,沒法像原生的字典一樣方便的通過暴露出來的api訪問數(shù)據(jù)景用。當(dāng)然也可以從頭開始把完整的Dictionary Interface
完全實(shí)現(xiàn)個(gè)遍涵叮,不過在每個(gè)自定義的類似的容器中都來一套肯定不行,這時(shí)候利用python內(nèi)置的UserDict.DictMixin就是一個(gè)不錯(cuò)的方式:
from UserDict import DictMixin
class BetterSimpleItemContainer(object, DictMixin):
def __getitem__(self, id):
return self.data[id]
def __setitem__(self, id, value):
self.data[id] = value
def __delitem__(self, id):
del self.data[id]
def keys(self):
return self.data.keys()
通過實(shí)現(xiàn)最小的Dictionary Interface
伞插,還有繼承DictMixin
實(shí)現(xiàn)Mixin模式割粮,我們就輕松獲得了完整的原生字典的行為:下表語法,get
, has_keys
, iteritems
, itervalues
甚至還有iterable protocol implementation等一系列的方法和實(shí)現(xiàn)媚污。
很多框架比如Django舀瓢, Django rest framework里面就普遍用到了Mixin這種模式,定義api或者viewset的時(shí)候就能夠通過多重繼承的方式服用一些功能
當(dāng)然耗美,Mixin模式也不能濫用京髓,至少他會(huì)污染你新定義的類,有時(shí)候還會(huì)帶來MRO的問題商架;不過把一些基礎(chǔ)和單一的功能比如一般希望通過interface/protocol
實(shí)現(xiàn)的功能放進(jìn)Mixin模塊里面還是不錯(cuò)的選擇:
class CommonEqualityMixin(object):
def __eq__(self, other):
return (isinstance(other, self.__class__)
and self.__dict__ == other.__dict__)
def __ne__(self, other):
return not self.__eq__(other)
class Foo(CommonEqualityMixin):
def __init__(self, item):
self.item = item
其實(shí)整個(gè)理解下來無非就是通過組合的方式獲得更多的功能,有點(diǎn)像C#堰怨, java里面的interface,強(qiáng)調(diào)“it can”的意思甸私,但相比起來簡單多了诚些,不需要顯示的約束,而且mixin模塊自帶實(shí)現(xiàn)皇型。在使用的時(shí)候一般把mixin的類放在父類的右邊似乎也是為了強(qiáng)調(diào)這并不是典型的多繼承诬烹,是一種特殊的多繼承
,而是在繼承了一個(gè)基類的基礎(chǔ)上弃鸦,順帶利用多重繼承的功能給這個(gè)子類添點(diǎn)料绞吁,增加一些其他的功能。保證Mixin的類功能單一具體唬格,混入之后家破,新的類的MRO樹其實(shí)也會(huì)相對很簡單颜说,并不會(huì)引起混亂。