許多有其他語言背景(比如C,C++等)的初學(xué)者都會(huì)以為在Python中,__init__方法充當(dāng)著類似與類構(gòu)造方法的角色。就大多數(shù)使用場景來看,這么理解似乎沒有什么問題:當(dāng)實(shí)例化一個(gè)對(duì)象的時(shí)候,使用my_class = MyClass(args)就會(huì)返回一個(gè)實(shí)例給my_class變量。
那實(shí)際上,在實(shí)例化一個(gè)類的時(shí)候,到底是一個(gè)怎樣的過程呢?首先要說的是,__init__并不是一個(gè)構(gòu)造方法,它的工作是在類的對(duì)象創(chuàng)建完成之后,對(duì)類的變量屬性進(jìn)行初始化。那類的對(duì)象又是由誰來創(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__沒有返回值,當(dāng)__new__返回類的對(duì)象之后,__init__方法才會(huì)執(zhí)行,若無對(duì)象返回,則__init__不會(huì)被調(diào)用。 - 二者的功能明顯不同:
__new__用于創(chuàng)建一個(gè)實(shí)例,__init__用于初始化一個(gè)實(shí)例 - 大多數(shù)情況下,我們需要重寫
__init__來實(shí)現(xiàn)實(shí)例對(duì)象的變量屬性初始化,而只有在子類繼承不可變類型的時(shí)候(比如:str,int,tuple等),才需要重寫__new__ - 在實(shí)現(xiàn)工廠模式、單例模式或者元類編程的時(shí)候,可能會(huì)需要顯式重寫
__new__來控制類的創(chuàng)建。 -
__metaclass__作用是類創(chuàng)建,__new__作用是實(shí)例創(chuàng)建,__init__作用是實(shí)例初始化。
最后,英文好的童鞋看這里