許多有其他語(yǔ)言背景(比如C,C++等)的初學(xué)者都會(huì)以為在Python
中阀参,__init__
方法充當(dāng)著類似與類構(gòu)造方法的角色酝润。就大多數(shù)使用場(chǎng)景來(lái)看吊说,這么理解似乎沒(méi)有什么問(wèn)題:當(dāng)實(shí)例化一個(gè)對(duì)象的時(shí)候卧秘,使用my_class = MyClass(args)就會(huì)返回一個(gè)實(shí)例給my_class變量绪撵。
那實(shí)際上葛超,在實(shí)例化一個(gè)類的時(shí)候暴氏,到底是一個(gè)怎樣的過(guò)程呢?首先要說(shuō)的是绣张,__init__
并不是一個(gè)構(gòu)造方法答渔,它的工作是在類的對(duì)象創(chuàng)建完成之后,對(duì)類的變量屬性進(jìn)行初始化侥涵。那類的對(duì)象又是由誰(shuí)來(lái)創(chuàng)建的呢沼撕?
答案是__new__
方法宋雏,它才是傳統(tǒng)意義上的構(gòu)造方法。
那__init__
和__new__
又有什么不同點(diǎn)呢务豺?
-
__init__
定義為object.__init__(self,*args)
磨总,而__new__
定義為object.__new__(cls,*args)
,由定義不難看出笼沥,__init__
是一個(gè)實(shí)例方法蚪燕,而__new__
是一個(gè)靜態(tài)方法。 -
__new__
會(huì)返回一個(gè)創(chuàng)建的實(shí)例奔浅,__init__
沒(méi)有返回值馆纳,當(dāng)__new__
返回類的對(duì)象之后,__init__
方法才會(huì)執(zhí)行汹桦,若無(wú)對(duì)象返回鲁驶,則__init__
不會(huì)被調(diào)用。 - 二者的功能明顯不同:
__new__
用于創(chuàng)建一個(gè)實(shí)例舞骆,__init__
用于初始化一個(gè)實(shí)例 - 大多數(shù)情況下钥弯,我們需要重寫
__init__
來(lái)實(shí)現(xiàn)實(shí)例對(duì)象的變量屬性初始化,而只有在子類繼承不可變類型的時(shí)候(比如:str,int,tuple等)督禽,才需要重寫__new__
- 在實(shí)現(xiàn)工廠模式寿羞、單例模式或者元類編程的時(shí)候,可能會(huì)需要顯式重寫
__new__
來(lái)控制類的創(chuàng)建赂蠢。 -
__metaclass__
作用是類創(chuàng)建绪穆,__new__
作用是實(shí)例創(chuàng)建,__init__
作用是實(shí)例初始化虱岂。
最后玖院,英文好的童鞋看這里