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
根據(jù)類創(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'>
# 根據(jù)類創(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中已經(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__()實例化方法,類在內存中創(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ù)據(jù)并繪制箱型圖和直方圖的方法實例
這篇文章主要給大家介紹了關于Python獲取excel的數(shù)據(jù)并繪制箱型圖和直方圖的相關資料,好的圖表能幫助我們深化數(shù)據(jù)的記憶點,文中通過圖文以及代碼示例將實現(xiàn)的方法介紹的非常詳細,需要的朋友可以參考下2023-12-12
python3實現(xiàn)網(wǎng)絡爬蟲之BeautifulSoup使用詳解
這篇文章主要介紹了python3實現(xiàn)網(wǎng)絡爬蟲之BeautifulSoup使用詳解,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-12-12
Python Opencv提取圖片中某種顏色組成的圖形的方法
這篇文章主要介紹了Python Opencv提取圖片中某種顏色組成的圖形的方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2019-09-09
Python?Requests使用Cookie的幾種方式詳解
這篇文章主要給大家介紹了關于Python?Requests使用Cookie的幾種方式,Python中的requests庫可以使用cookie來維持會話狀態(tài),實現(xiàn)登錄等操作,需要的朋友可以參考下2023-07-07

