MetaClass元類,本質(zhì)也是一個(gè)類,但和普通類的用法不同,它可以對(duì)類內(nèi)部的定義(包括類屬性和類方法)進(jìn)行動(dòng)態(tài)的修改??梢赃@么說(shuō),使用元類的主要目的就是為了實(shí)現(xiàn)在創(chuàng)建類時(shí),能夠動(dòng)態(tài)地改變類中定義的屬性或者方法。
不要從字面上去理解元類的含義,事實(shí)上 MetaClass 中的 Meta 這個(gè)詞根,起源于希臘語(yǔ)詞匯 meta,包含“超越”和“改變”的意思。
舉個(gè)例子,根據(jù)實(shí)際場(chǎng)景的需要,我們要為多個(gè)類添加一個(gè) name 屬性和一個(gè) say() 方法。顯然有多種方法可以實(shí)現(xiàn),但其中一種方法就是使用 MetaClass 元類。
如果在創(chuàng)建類時(shí),想用 MetaClass 元類動(dòng)態(tài)地修改內(nèi)部的屬性或者方法,則類的創(chuàng)建過(guò)程將變得復(fù)雜:先創(chuàng)建 MetaClass 元類,然后用元類去創(chuàng)建類,最后使用該類的實(shí)例化對(duì)象實(shí)現(xiàn)功能。
和前面章節(jié)創(chuàng)建的類不同,如果想把一個(gè)類設(shè)計(jì)成 MetaClass 元類,其必須符合以下條件:
必須顯式繼承自 type 類;
類中需要定義并實(shí)現(xiàn) __new__() 方法,該方法一定要返回該類的一個(gè)實(shí)例對(duì)象,因?yàn)樵谑褂迷悇?chuàng)建類時(shí),該 __new__() 方法會(huì)自動(dòng)被執(zhí)行,用來(lái)修改新建的類。
講了這么多,讀者可能對(duì) MetaClass 元類的功能還是比較懵懂。沒(méi)關(guān)系,我們先嘗試定義一個(gè) MetaClass 元類:
此程序中,首先可以斷定 FirstMetaClass 是一個(gè)類。其次,由于該類繼承自 type 類,并且內(nèi)部實(shí)現(xiàn)了 __new__() 方法,因此可以斷定 FirstMetaCLass 是一個(gè)元類。
可以看到,在這個(gè)元類的 __new__() 方法中,手動(dòng)添加了一個(gè) name 屬性和 say() 方法。這意味著,通過(guò) FirstMetaClass 元類創(chuàng)建的類,會(huì)額外添加 name 屬性和 say() 方法。通過(guò)如下代碼,可以驗(yàn)證這個(gè)結(jié)論:
可以看到,在創(chuàng)建類時(shí),通過(guò)在標(biāo)注父類的同時(shí)指定元類(格式為metaclass=元類名),則當(dāng) Python 解釋器在創(chuàng)建這該類時(shí),F(xiàn)irstMetaClass 元類中的 __new__ 方法就會(huì)被調(diào)用,從而實(shí)現(xiàn)動(dòng)態(tài)修改類屬性或者類方法的目的。
運(yùn)行上面的程序,輸出結(jié)果為:
顯然,F(xiàn)irstMetaClass 元類的 __new__() 方法動(dòng)態(tài)地為 Clanguage 類添加了 name 屬性和 say() 方法,因此,即便該類在定義時(shí)是空類,它也依然有 name 屬性和 say() 方法。
對(duì)于 MetaClass 元類,它多用于創(chuàng)建 API,因此我們幾乎不會(huì)使用到它。