Python Metaclass原理與實(shí)現(xiàn)過(guò)程詳細(xì)講解
前言
學(xué)習(xí)python時(shí),對(duì) meta class 并不陌生,django 的model, form定義中經(jīng)常會(huì)遇到class Meta 的代碼, 但 django中沒(méi)有介紹模型與表單中的 meta 屬性是怎么回事。 python的眾多教程,也很少有作者提到,或把meta class 講清楚,造成大家以為metaclass 很難理解。 今天我嘗試用實(shí)例代碼的方式,讓metaclass 更容易理解與使用。
class Person(models.Model):
cname = models.CharField(max_length=30, verbose_name="姓名")
user = models.ForeignKey(
get_user_model(), null=True, on_delete=models.CASCADE)
birth_date = models.DateField(verbose_name="出生日期")
def __str__(self):
return self.cname
class Meta:
verbose_name = "人員信息"
db_table = "tbl_person"Python與java 一樣,宣稱自己一切皆對(duì)象,且有過(guò)之而不及。python的數(shù)據(jù)類型是類對(duì)象,甚至函數(shù)本身也是對(duì)象。 Python object 是所有類型的基類,各種數(shù)據(jù)類型, 函數(shù), class 都是object 的派生類,因此,從object的概念上推理,存在某些 class ,其可以生成其它c(diǎn)lass, 這種 class 就稱為meta class., 也稱元類.
本文將討論meta class的定義,以及使用方式。 內(nèi)容共包含3個(gè)主題:
- 用 type 定義類
- 如何編寫(xiě) Metaclass
- Metaclass 的實(shí)際應(yīng)用
Type 來(lái)定義類
1個(gè) class 也是1個(gè)object, 其包含若干屬性,以及一些方法。 type () 通常用來(lái)顯示 變量與函數(shù)的類型,但也可以用type 來(lái)創(chuàng)建 class. 下面我們來(lái)演示這一過(guò)程。
首先,先用標(biāo)準(zhǔn)的方式,class關(guān)鍵字來(lái)創(chuàng)建1個(gè) Food 類。
class Food(object):
def __init__(self, food):
self.food = food
def get_food(self):
return self.food
def main():
myfood = Food(food='蛋炒飯')
print(myfood.get_food())
main()output
蛋炒飯
下面演示如何用 type來(lái)創(chuàng)建1個(gè)Food 類
def init(self, food):
self.food = food
def get_food(self):
return self.food
Food = type("Food", (object,), {
'__init__': init,
'get_food': get_food,
})
myfood = Food(food="蛋炒飯")
print(myfood.get_food())output:
蛋炒飯
要?jiǎng)?chuàng)建一個(gè)class對(duì)象,type()函數(shù)依次傳入3個(gè)參數(shù):
- 第1個(gè)參數(shù)是string類型,class的名稱;
- 第2個(gè)參數(shù)是1個(gè)tuple , (object, ) , 指定繼承的父類是 object,注意支持多重繼承,如果只有一個(gè)父類,后面是加個(gè) , 號(hào);
- 第3個(gè)參數(shù)是 dictionary 類型, 指定 class的屬性與方法名稱,這里我們把函數(shù)init綁定到方法名
__init__上
通過(guò)type()函數(shù)創(chuàng)建的類和直接寫(xiě)class是完全一樣的,因?yàn)镻ython解釋器遇到class定義時(shí),僅僅是掃描一下class定義的語(yǔ)法,然后調(diào)用type()函數(shù)創(chuàng)建出class
用 type來(lái)增加1個(gè)子類VegFood, 其父類為Food
def init(self, food):
self.food = food
def get_food(self):
return self.food
Food = type("Food", (object,), {
'__init__': init,
'get_food': get_food,
})
def get_vegfood(self):
return {'清炒菠菜', '干煸豆角'}
# 新建子類,父類為Food
VegFood = type("VegFood", (Food,), {
"get_vegfood": get_vegfood,
})
veg = VegFood(food="蛋炒飯")
print(veg.get_food())
print(veg.get_vegfood())output:
蛋炒飯
{'干煸豆角', '清炒菠菜'}
可以看到,子類 VegFood可以繼承 父類Food的方法get_food(),也可以使用子類的方法 get_vegfood()
編寫(xiě) meta class
metaclass,直譯為元類,繼承自type. 因此,如前一節(jié)的示例 ,metaclass 也可以用來(lái)創(chuàng)建類。
用metaclass后, 關(guān)于類的編程過(guò)程: 先定義metaclass,就可以創(chuàng)建類,最后創(chuàng)建實(shí)例。
所以,metaclass允許你創(chuàng)建類或者修改類。換句話說(shuō),你可以把類看成是metaclass創(chuàng)建出來(lái)的“實(shí)例”。
metaclass的定義過(guò)程,主要是實(shí)現(xiàn) __new__ 方法,有4個(gè)傳入?yún)?shù), 與 type創(chuàng)建類的參數(shù)相似
- metaclass 自身
- 類名
- 父類( tuple 類型)
- 類的屬性與方法字典
metaclass還可以實(shí)現(xiàn)1個(gè) __call__方法,在該類被調(diào)用時(shí)執(zhí)行此方法。
metacalss 示例 :
class MetaCls(type):
"""metaclass的示例"""
def __new__(cls, clsname, superclasses, attributedict):
print("clsname:", clsname)
print("superclasses:", superclasses)
print("attrdict:", attributedict)
return super(MetaCls, cls).__new__(cls,
clsname, superclasses, attributedict)
def demo(self):
print("Info from demo method")
# generate a class, Example
Example = MetaCls('Example', (object, ), {"demo": demo, })
Example.clsname = "ExampleClass" # 修改 metaclass的屬性值
print("class type:", type(Example))
myexample = Example() # 新建1個(gè)Example的實(shí)例對(duì)象
print(myexample.clsname) # 打印父類屬性值
myexample.demo() # 打印子類方法output
clsname: Example
superclasses: (<class 'object'>,)
attrdict: {'demo': <function demo at 0x00000169DF2CC280>}
class type: <class '__main__.MetaCls'>
ExampleClass
Info from demo method
從上例 可以看到, Example類的生成方式,就是 MetaCls類的實(shí)例 化過(guò)程,example可以添加自己的方法,可以修改從父類繼承的屬性。
metaclass應(yīng)用
實(shí)際開(kāi)發(fā)中遇到的metaclass是django中定義model, form時(shí),可以定義metaclass的屬性,其實(shí)就是django的ORM使用了 metaclass的編程方式,由于其實(shí)現(xiàn)較復(fù)雜,就不深入討論。
這里我們來(lái)看1個(gè)更簡(jiǎn)單,而且非常實(shí)用的metaclass的例子: 用metaclass實(shí)現(xiàn)單例模式。
單例模式提供了這樣一個(gè)機(jī)制,即確保類有且只有一個(gè)特定類型的對(duì)象,并提供全局 訪問(wèn)點(diǎn)。因此,單例模式通常用于下列情形,例如日志記錄或數(shù)據(jù)庫(kù)操作、打印機(jī)后臺(tái)處 理程序,以及其他程序—該程序運(yùn)行過(guò)程中只能生成一個(gè)實(shí)例,以避免對(duì)同一資源產(chǎn)生 相互沖突的請(qǐng)求。
class SingletonMeta(type):
_instances = {}
def __call__(cls, *args, **kwargs):
if cls not in cls._instances:
cls._instances[cls] = super(
SingletonMeta, cls).__call__(*args, **kwargs)
return cls._instances[cls]
class SingletonClass(metaclass=SingletonMeta):
pass
s = SingletonClass()
print("Object created", s)
s1 = SingletonClass()
print("Object created", s1)output :
Object created <__main__.SingletonClass object at 0x0000020CF1F20130>
Object created <__main__.SingletonClass object at 0x0000020CF1F20130>
可以看到,SingletonMeta用metaclass的方法定義了1個(gè)單例類, SingletonClass繼承自SingletonMeta, s 與 s1 雖然分別實(shí)例化,但內(nèi)存中的地址都指向同1個(gè)SingletonClass對(duì)象,這樣就實(shí)現(xiàn)了單例化。
而且metaclass實(shí)現(xiàn)單例模式,比 object方式實(shí)現(xiàn)的單例化有明顯的優(yōu)勢(shì),任何類都可以通過(guò)繼承SingletonMeta實(shí)現(xiàn)單例模式, 是不是很酷.
到此這篇關(guān)于Python Metaclass原理與實(shí)現(xiàn)過(guò)程詳細(xì)講解的文章就介紹到這了,更多相關(guān)Python Metaclass內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
如何使用selenium和requests組合實(shí)現(xiàn)登錄頁(yè)面
這篇文章主要介紹了如何使用selenium和requests組合實(shí)現(xiàn)登錄頁(yè)面,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-02-02
三種Matplotlib中動(dòng)態(tài)更新繪圖的方法總結(jié)
這篇文章主要為大家詳細(xì)介紹了如何隨著數(shù)據(jù)的變化動(dòng)態(tài)更新Matplotlib(Python的數(shù)據(jù)可視化庫(kù))圖,文中介紹了常用的三種方法,希望對(duì)大家有所幫助2024-04-04
Django的開(kāi)發(fā)步驟原來(lái)是這樣的
這篇文章主要為大家詳細(xì)介紹了Django的開(kāi)發(fā)步驟,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助2022-02-02
python實(shí)現(xiàn)從本地?cái)z像頭和網(wǎng)絡(luò)攝像頭截取圖片功能
這篇文章主要介紹了python實(shí)現(xiàn)從本地?cái)z像頭和網(wǎng)絡(luò)攝像頭截取圖片功能 ,文中給大家提到了python , opencv 打開(kāi)網(wǎng)絡(luò)攝像頭讀取圖像的實(shí)現(xiàn)代碼,需要的朋友可以參考下2019-07-07
pygame實(shí)現(xiàn)鍵盤的連續(xù)監(jiān)控
這篇文章主要為大家詳細(xì)介紹了pygame實(shí)現(xiàn)鍵盤的連續(xù)監(jiān)控,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-04-04
Python實(shí)現(xiàn)希爾排序,歸并排序和桶排序的示例代碼
希爾、歸并、快速排序算法可歸為同一類,它們的共同點(diǎn)都是建立在分治思想之上。把大問(wèn)題分拆成小問(wèn)題,解決所有小問(wèn)題后,再合并每一個(gè)小問(wèn)題的結(jié)果,最終得到對(duì)原始問(wèn)題的解答。本文將介紹這三種算法的實(shí)現(xiàn)代碼,需要的可以參考一下2022-04-04

