python高級之元類的用法總結(jié)
一、Type創(chuàng)建類
class A(object): def __init__(self, name): self.name = name def __new__(cls, *args, **kwargs): data = object.__new__(cls) return data
根據(jù)類創(chuàng)建對象
obj=A(‘kobe’)
1、執(zhí)行類的new方法,創(chuàng)建對象(空對象),【構(gòu)造方法】 {}
2、執(zhí)行類的init方法,初始化對象 ,【初始化方法】 {‘name’:‘kobe’}
對象是基于類創(chuàng)建出來的;問題:類是由誰創(chuàng)建的?
類默認是由type創(chuàng)建的
1、傳統(tǒng)方式創(chuàng)建類
class A(object): v1 = 123 def func(self): return 666 print(A) #<class '__main__.A'>
2、非傳統(tǒng)方式
類名
繼承類
成員
A1 = type('A', (object,), {"v1": 123, "func": lambda self: 666}) print(A1) #<class '__main__.A'> # 根據(jù)類創(chuàng)建對象 obj1 = A1() #<__main__.A object at 0x0000017049D035E0> print(obj1) # 調(diào)用對象中的func方法 print(obj1.func()) #666
類默認由type創(chuàng)建,怎么讓一個類的創(chuàng)建改為其他的東西呢? 元類。
二、元類
元類:指定類由誰來創(chuàng)建
1、Foo類由MyType創(chuàng)建,代碼如下
class MyType(type): pass # Foo類由MyType創(chuàng)建 class Foo(object, metaclass=MyType): pass
2、假設(shè)Foo是一個對象,它是由MyType類創(chuàng)建。
class MyType(type): def __init__(self, *args, **kwargs): print('init') super().__init__(*args, **kwargs) def __new__(cls, *args, **kwargs): # 創(chuàng)建類 print('new') new_cls = super().__new__(cls, *args, **kwargs) print(new_cls) return new_cls # 假設(shè)Foo是一個對象,由MyType類創(chuàng)建。 class Foo(object, metaclass=MyType): pass
執(zhí)行結(jié)果
new
<class '__main__.Foo'>
init
3、類加括號得到的是對象,執(zhí)行的是__init__()和__new__()
方法,對象加括號執(zhí)行__call__()
方法
class F1(object): def __init__(self): print('init') # def __new__(cls, *args, **kwargs): # print('new') def __call__(self, *args, **kwargs): print(111) # 類加括號得到的是對象,執(zhí)行的是__init__和__new__方法 obj = F1() # 對象加括號執(zhí)行call方法 obj()
執(zhí)行結(jié)果
init
111
4、假設(shè)Foo是一個對象,由MyType創(chuàng)建。
Foo類其實是MyType的一個對象。
Foo() 加括號執(zhí)行的是——》MyType類中的__call__()
方法
__call__()
方法實現(xiàn)的功能是:
1、調(diào)用自己那個類的__new__
方法創(chuàng)建對象
empty_object=self.__new__(self)
2、調(diào)用你自己那個類__init__
方法去初始化對象
self.__init__(empty_object,*args,**kwargs)
3、將創(chuàng)建的對象返回:return empty_object
class MyType(type): def __init__(self, *args, **kwargs): print('init') super().__init__(*args, **kwargs) def __new__(cls, *args, **kwargs): # 創(chuàng)建類 print('new') new_cls = super().__new__(cls, *args, **kwargs) print(new_cls) return new_cls def __call__(self, *args, **kwargs): # 1、調(diào)用自己哪個類的__new__方法創(chuàng)建對象 empty_object=self.__new__(self) #2、調(diào)用你自己那個類 __init__方法去初始化 self.__init__(empty_object,*args,**kwargs) return empty_object # 假設(shè)Foo是一個對象,由MyType創(chuàng)建。 # Foo類其實是MyType的一個對象。 # Foo() ——》MyType對象() class Foo(object, metaclass=MyType): def __init__(self,name): self.name=name v1=Foo('666') print(v1) print(v1.name)
三、總結(jié)
1、當我們不寫MyType類時,Type中已經(jīng)幫我們定義好了__init____new____call__
方法了
當我定義了MyType類時,__init____new____call__
方法是我們自己定義的
2、這就是為什么實例化對象的時候,先去創(chuàng)建對象再去初始化對象,Type類中已經(jīng)實現(xiàn)了先 __new__()
再__init__()
3、代碼從上到下執(zhí)行,當執(zhí)行到class Foo(object, metaclass=MyType)
時
先創(chuàng)建這個類,去MyType中的__new__()
創(chuàng)建方法
__init__()
實例化方法,類在內(nèi)存中創(chuàng)建好了,
但是MyType中的__call__()
方法是不執(zhí)行的,因為類后面沒有加括號
如果類后面加了括號Foo(),會執(zhí)行MyType的__call__()
方法
4、如果自己定義的類中實現(xiàn)了__call__
方法,此時是不會執(zhí)行的,因為Foo是MyType創(chuàng)建出來的
5、如果要執(zhí)行Foo類的__call__
方法,需要實例化Foo類的對象v1,
然后v1加括號
附:元類的最佳實踐和注意事項
最佳實踐
僅在必要時使用元類。元類是高級編程工具,通常不需要在日常編程中使用。
考慮繼承自type以定義元類,因為type是Python中的內(nèi)置元類。
在元類的__new__方法中,要返回一個類對象,通常是使用super().__new__來創(chuàng)建它。
注意事項
元類可以控制類的創(chuàng)建和初始化,但要小心不要過度使用,以免使代碼變得復(fù)雜和難以理解。
在元類中的操作可能會影響所有使用該元類創(chuàng)建的類,因此要小心不要引入意外的副作用。
元類的概念可能對初學(xué)者來說有點復(fù)雜,建議在熟悉Python的基礎(chǔ)之后再深入學(xué)習元類。
總結(jié)
到此這篇關(guān)于python高級之元類用法總結(jié)的文章就介紹到這了,更多相關(guān)python高級之元類內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Python獲取excel的數(shù)據(jù)并繪制箱型圖和直方圖的方法實例
這篇文章主要給大家介紹了關(guān)于Python獲取excel的數(shù)據(jù)并繪制箱型圖和直方圖的相關(guān)資料,好的圖表能幫助我們深化數(shù)據(jù)的記憶點,文中通過圖文以及代碼示例將實現(xiàn)的方法介紹的非常詳細,需要的朋友可以參考下2023-12-12python3實現(xiàn)網(wǎng)絡(luò)爬蟲之BeautifulSoup使用詳解
這篇文章主要介紹了python3實現(xiàn)網(wǎng)絡(luò)爬蟲之BeautifulSoup使用詳解,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-12-12Python Opencv提取圖片中某種顏色組成的圖形的方法
這篇文章主要介紹了Python Opencv提取圖片中某種顏色組成的圖形的方法,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習或者工作具有一定的參考學(xué)習價值,需要的朋友們下面隨著小編來一起學(xué)習學(xué)習吧2019-09-09Python?Requests使用Cookie的幾種方式詳解
這篇文章主要給大家介紹了關(guān)于Python?Requests使用Cookie的幾種方式,Python中的requests庫可以使用cookie來維持會話狀態(tài),實現(xiàn)登錄等操作,需要的朋友可以參考下2023-07-07python進行數(shù)據(jù)預(yù)處理的4個重要步驟
在數(shù)據(jù)科學(xué)項目中,數(shù)據(jù)預(yù)處理是最重要的事情之一,本文詳細給大家介紹python進行數(shù)據(jù)預(yù)處理的4個重要步驟:拆分訓(xùn)練集和測試集,處理缺失值,處理分類特征和進行標準化處理,需要的朋友可以參考下2023-06-06