python高級之元類的用法總結
一、Type創(chuàng)建類
class A(object): def __init__(self, name): self.name = name def __new__(cls, *args, **kwargs): data = object.__new__(cls) return data
根據類創(chuàng)建對象
obj=A(‘kobe’)
1、執(zhí)行類的new方法,創(chuàng)建對象(空對象),【構造方法】 {}
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'> # 根據類創(chuàng)建對象 obj1 = A1() #<__main__.A object at 0x0000017049D035E0> print(obj1) # 調用對象中的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、假設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 # 假設Foo是一個對象,由MyType類創(chuàng)建。 class Foo(object, metaclass=MyType): pass
執(zhí)行結果
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í)行結果
init
111
4、假設Foo是一個對象,由MyType創(chuàng)建。
Foo類其實是MyType的一個對象。
Foo() 加括號執(zhí)行的是——》MyType類中的__call__()
方法
__call__()
方法實現(xiàn)的功能是:
1、調用自己那個類的__new__
方法創(chuàng)建對象
empty_object=self.__new__(self)
2、調用你自己那個類__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、調用自己哪個類的__new__方法創(chuàng)建對象 empty_object=self.__new__(self) #2、調用你自己那個類 __init__方法去初始化 self.__init__(empty_object,*args,**kwargs) return empty_object # 假設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)
三、總結
1、當我們不寫MyType類時,Type中已經幫我們定義好了__init____new____call__
方法了
當我定義了MyType類時,__init____new____call__
方法是我們自己定義的
2、這就是為什么實例化對象的時候,先去創(chuàng)建對象再去初始化對象,Type類中已經實現(xiàn)了先 __new__()
再__init__()
3、代碼從上到下執(zhí)行,當執(zhí)行到class Foo(object, metaclass=MyType)
時
先創(chuàng)建這個類,去MyType中的__new__()
創(chuàng)建方法
__init__()
實例化方法,類在內存中創(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中的內置元類。
在元類的__new__方法中,要返回一個類對象,通常是使用super().__new__來創(chuàng)建它。
注意事項
元類可以控制類的創(chuàng)建和初始化,但要小心不要過度使用,以免使代碼變得復雜和難以理解。
在元類中的操作可能會影響所有使用該元類創(chuàng)建的類,因此要小心不要引入意外的副作用。
元類的概念可能對初學者來說有點復雜,建議在熟悉Python的基礎之后再深入學習元類。
總結
到此這篇關于python高級之元類用法總結的文章就介紹到這了,更多相關python高級之元類內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
Python獲取excel的數(shù)據并繪制箱型圖和直方圖的方法實例
這篇文章主要給大家介紹了關于Python獲取excel的數(shù)據并繪制箱型圖和直方圖的相關資料,好的圖表能幫助我們深化數(shù)據的記憶點,文中通過圖文以及代碼示例將實現(xiàn)的方法介紹的非常詳細,需要的朋友可以參考下2023-12-12python3實現(xiàn)網絡爬蟲之BeautifulSoup使用詳解
這篇文章主要介紹了python3實現(xiàn)網絡爬蟲之BeautifulSoup使用詳解,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-12-12Python Opencv提取圖片中某種顏色組成的圖形的方法
這篇文章主要介紹了Python Opencv提取圖片中某種顏色組成的圖形的方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2019-09-09Python?Requests使用Cookie的幾種方式詳解
這篇文章主要給大家介紹了關于Python?Requests使用Cookie的幾種方式,Python中的requests庫可以使用cookie來維持會話狀態(tài),實現(xiàn)登錄等操作,需要的朋友可以參考下2023-07-07