__new__: 對(duì)象的創(chuàng)建,是一個(gè)靜態(tài)方法,第一個(gè)參數(shù)是cls。(想想也是,不可能是self,對(duì)象還沒創(chuàng)建,哪來的self)
在梅州等地區(qū),都構(gòu)建了全面的區(qū)域性戰(zhàn)略布局,加強(qiáng)發(fā)展的系統(tǒng)性、市場(chǎng)前瞻性、產(chǎn)品創(chuàng)新能力,以專注、極致的服務(wù)理念,為客戶提供成都做網(wǎng)站、成都網(wǎng)站制作 網(wǎng)站設(shè)計(jì)制作按需定制,公司網(wǎng)站建設(shè),企業(yè)網(wǎng)站建設(shè),品牌網(wǎng)站制作,成都營(yíng)銷網(wǎng)站建設(shè),外貿(mào)網(wǎng)站建設(shè),梅州網(wǎng)站建設(shè)費(fèi)用合理。
__init__ : 對(duì)象的初始化, 是一個(gè)實(shí)例方法,第一個(gè)參數(shù)是self。
__call__ : 對(duì)象可call,注意不是類,是對(duì)象。
先有創(chuàng)建,才有初始化。即先__new__,而后__init__。
上面說的不好理解,看例子。
1.對(duì)于__new__
可以看到,輸出來是一個(gè)Bar對(duì)象。
__new__方法在類定義中不是必須寫的,如果沒定義,默認(rèn)會(huì)調(diào)用object.__new__去創(chuàng)建一個(gè)對(duì)象。如果定義了,就是override,可以custom創(chuàng)建對(duì)象的行為。
聰明的讀者可能想到,既然__new__可以custom對(duì)象的創(chuàng)建,那我在這里做一下手腳,每次創(chuàng)建對(duì)象都返回同一個(gè),那不就是單例模式了嗎?沒錯(cuò),就是這樣??梢杂^摩《飄逸的python - 單例模式亂彈》
定義單例模式時(shí),因?yàn)樽远x的__new__重載了父類的__new__,所以要自己顯式調(diào)用父類的__new__,即object.__new__(cls, *args, **kwargs),或者用super()。,不然就不是extend原來的實(shí)例了,而是替換原來的實(shí)例。
2.對(duì)于__init__
使用Python寫過面向?qū)ο蟮拇a的同學(xué),可能對(duì) __init__ 方法已經(jīng)非常熟悉了,__init__ 方法通常用在初始化一個(gè)類實(shí)例的時(shí)候。例如:
這樣便是__init__最普通的用法了。但__init__其實(shí)不是實(shí)例化一個(gè)類的時(shí)候第一個(gè)被調(diào)用 的方法。當(dāng)使用 Persion(name, age) 這樣的表達(dá)式來實(shí)例化一個(gè)類時(shí),最先被調(diào)用的方法 其實(shí)是 __new__ 方法。
3.對(duì)于__call__
對(duì)象通過提供__call__(slef, [,*args [,**kwargs]])方法可以模擬函數(shù)的行為,如果一個(gè)對(duì)象x提供了該方法,就可以像函數(shù)一樣使用它,也就是說x(arg1, arg2...) 等同于調(diào)用x.__call__(self, arg1, arg2) 。模擬函數(shù)的對(duì)象可以用于創(chuàng)建防函數(shù)(functor) 或代理(proxy).
總結(jié),在Python中,類的行為就是這樣,__new__、__init__、__call__等方法不是必須寫的,會(huì)默認(rèn)調(diào)用,如果自己定義了,就是override,可以custom。既然override了,通常也會(huì)顯式調(diào)用進(jìn)行補(bǔ)償以達(dá)到extend的目的。
這也是為什么會(huì)出現(xiàn)"明明定義def _init__(self, *args, **kwargs),對(duì)象怎么不進(jìn)行初始化"這種看起來詭異的行為。(注,這里_init__少寫了個(gè)下劃線,因?yàn)開_init__不是必須寫的,所以這里不會(huì)報(bào)錯(cuò),而是當(dāng)做一個(gè)新的方法_init__)
__call__
在Python中,函數(shù)其實(shí)是一個(gè)對(duì)象:
f = abs
f.__name__
'abs'
f(-123)
由于 f 可以被調(diào)用,所以,f 被稱為可調(diào)用對(duì)象。
所有的函數(shù)都是可調(diào)用對(duì)象。
一個(gè)類實(shí)例也可以變成一個(gè)可調(diào)用對(duì)象,只需要實(shí)現(xiàn)一個(gè)特殊方法__call__()。
我們把 Person 類變成一個(gè)可調(diào)用對(duì)象:
class Person(object):
def __init__(self, name, gender):
self.name = name
self.gender = gender
def __call__(self, friend):
print 'My name is %s...' % self.name
print 'My friend is %s...' % friend
現(xiàn)在可以對(duì) Person 實(shí)例直接調(diào)用:
p = Person('Bob', 'male')
p('Tim')
My name is Bob...
My friend is Tim...
單看 p('Tim') 你無法確定 p 是一個(gè)函數(shù)還是一個(gè)類實(shí)例,所以,在Python中,函數(shù)也是對(duì)象,對(duì)象和函數(shù)的區(qū)別并不顯著。
任務(wù)
改進(jìn)一下前面定義的斐波那契數(shù)列:
class Fib(object):
???
請(qǐng)加一個(gè)__call__方法,讓調(diào)用更簡(jiǎn)單:
f = Fib()
print f(10)
[0, 1, 1, 2, 3, 5, 8, 13, 21, 34]
魔法方法 (Magic Methods) 是Python中的內(nèi)置函數(shù),一般以雙下劃線開頭和結(jié)尾,例如__ init__ 、 __del__ 等。之所以稱之為魔法方法,是因?yàn)檫@些方法會(huì)在進(jìn)行特定的操作時(shí)會(huì)自動(dòng)被調(diào)用。
在Python中,可以通過dir()方法來查看某個(gè)對(duì)象的所有方法和屬性,其中雙下劃線開頭和結(jié)尾的就是該對(duì)象的魔法方法。以字符串對(duì)象為例:
可以看到字符串對(duì)象有 __add__ 方法,所以在Python中可以直接對(duì)字符串對(duì)象使用"+"操作,當(dāng)Python識(shí)別到"+"操作時(shí),就會(huì)調(diào)用該對(duì)象的 __add__ 方法。有需要時(shí)我們可以在自己的類中重寫 __add__ 方法來完成自己想要的效果。
我們重寫了 __add__ 方法,當(dāng)Python識(shí)別"+"操作時(shí),會(huì)自動(dòng)調(diào)用重寫后的 __add__ 方法??梢钥吹?,魔法方法在類或?qū)ο蟮哪承┦录霭l(fā)后會(huì)自動(dòng)執(zhí)行,如果希望根據(jù)自己的程序定制特殊功能的類,那么就需要對(duì)這些方法進(jìn)行重寫。使用魔法方法,我們可以非常方便地給類添加特殊的功能。
1、構(gòu)造與初始化
__ new __ 、 __ init __ 這兩個(gè)魔法方法常用于對(duì)類的初始化操作。上面我們創(chuàng)建a1 = A("hello")時(shí),但首先調(diào)用的是 __ new __ ;初始化一個(gè)類分為兩步:
a.調(diào)用該類的new方法,返回該類的實(shí)例對(duì)象
b.調(diào)用該類的init方法,對(duì)實(shí)例對(duì)象進(jìn)行初始化。
__new__ (cls, *args, **kwargs)至少需要一個(gè)cls參數(shù),代表傳入的類。后面兩個(gè)參數(shù)傳遞給 __ init __ 。在 __ new __ 可以決定是否繼續(xù)調(diào)用 __ init __ 方法,只有當(dāng) __ new __ 返回了當(dāng)前類cls的實(shí)例,才會(huì)接著調(diào)用 __ init __ 。結(jié)合 __ new __ 方法的特性,我們可以通過重寫 __ new __ 方法實(shí)現(xiàn)Python的單例模式:
可以看到雖然創(chuàng)建了兩個(gè)對(duì)象,但兩個(gè)對(duì)象的地址相同。
2、控制屬性訪問這類魔法
方法主要對(duì)對(duì)象的屬性進(jìn)行訪問、定義、修改時(shí)起作用。主要有:
__getattr__(self, name): 定義當(dāng)用戶試圖獲取一個(gè)屬性時(shí)的行為。
__getattribute__(self, name):定義當(dāng)該類的屬性被訪問時(shí)的行為(先調(diào)用該方法,查看是否存在該屬性,若不存在,接著去調(diào)用getattr)。
__setattr__(self, name, value):定義當(dāng)一個(gè)屬性被設(shè)置時(shí)的行為。
當(dāng)初始化屬性時(shí)如self.a=a時(shí)或修改實(shí)例屬性如ins.a=1時(shí)本質(zhì)時(shí)調(diào)用魔法方法self. __ setattr __ (name,values);當(dāng)實(shí)例訪問某個(gè)屬性如ins.a本質(zhì)是調(diào)用魔法方法a. __ getattr __ (name)
3、容器類操作
有一些方法可以讓我們自己定義自己的容器,就像Python內(nèi)置的List,Tuple,Dict等等;容器分為可變?nèi)萜骱筒豢勺內(nèi)萜鳌?/p>
如果自定義一個(gè)不可變?nèi)萜鞯脑挘荒芏x__ len__ 和__ getitem__ ;定義一個(gè)可變?nèi)萜鞒瞬豢勺內(nèi)萜鞯乃心Хǚ椒?,還需要定義__ setitem__ 和__ delitem__ ;如果容器可迭代。還需要定義__ iter __。
__len__(self):返回容器的長(zhǎng)度
__getitem__(self,key):當(dāng)需要執(zhí)行self[key]的方式去調(diào)用容器中的對(duì)象,調(diào)用的是該方法
__setitem__(self,key,value):當(dāng)需要執(zhí)行self[key] = value時(shí),調(diào)用的是該方法
__iter__(self):當(dāng)容器可以執(zhí)行 for x in container:,或者使用iter(container)時(shí),需要定義該方法
下面舉一個(gè)例子,實(shí)現(xiàn)一個(gè)容器,該容器有List的一般功能,同時(shí)增加一些其它功能如訪問第一個(gè)元素,最后一個(gè)元素,記錄每個(gè)元素被訪問的次數(shù)等。
這類方法的使用場(chǎng)景主要在你需要定義一個(gè)滿足需求的容器類數(shù)據(jù)結(jié)構(gòu)時(shí)會(huì)用到,比如可以嘗試自定義實(shí)現(xiàn)樹結(jié)構(gòu)、鏈表等數(shù)據(jù)結(jié)構(gòu)(在collections中均已有),或者項(xiàng)目中需要定制的一些容器類型。
魔法方法在Python代碼中能夠簡(jiǎn)化代碼,提高代碼可讀性,在常見的Python第三方庫(kù)中可以看到很多對(duì)于魔法方法的運(yùn)用。
因此當(dāng)前這篇文章僅是拋磚引玉,真正的使用需要在開源的優(yōu)秀源碼中以及自身的工程實(shí)踐中不斷加深理解并合適應(yīng)用。
1、 定義一個(gè)特殊的 __slots__ 變量,來限制該class實(shí)例能添加的屬性
2、 內(nèi)置的 @property(關(guān)鍵字) 裝飾器就是負(fù)責(zé)把一個(gè)方法變成屬性調(diào)用的。@property.setter(這里的property是類里面的屬性名)負(fù)責(zé)把一個(gè)setter方法變成屬性賦值。
3、 __str__(),__repr__(),__iter__(),__next__(),__getitem__(),__setitem__(),__delitem__(),__getattr__(),__call__()
Python中如何實(shí)現(xiàn)運(yùn)算符的重載,即實(shí)現(xiàn)例如a+b這樣的運(yùn)算符操作呢?
在C++中可以使用 operator 關(guān)鍵字實(shí)現(xiàn)運(yùn)算符的重載。但是在Python中沒有類似這樣的關(guān)鍵字,所以要實(shí)現(xiàn)運(yùn)算符的重載,就要用到Python的魔法函數(shù)。Python魔法函數(shù)是以雙下劃線開頭,雙下劃線結(jié)尾的一組函數(shù)。我們?cè)陬惗x中最常用到的 __init__ 函數(shù)就是這樣一個(gè)魔法函數(shù),它在創(chuàng)建類對(duì)象時(shí)被自動(dòng)調(diào)用。
下面我們來看個(gè)簡(jiǎn)單的例子。
上述代碼示例了幾個(gè)魔法函數(shù)的用法。 __add__ 函數(shù)對(duì)應(yīng)了二元運(yùn)算符+,當(dāng)執(zhí)行a+b語(yǔ)句時(shí),python就會(huì)自動(dòng)調(diào)用a. add (b)。 對(duì)于上述例子中的v1+v2+v3,則相當(dāng)于調(diào)用了(v1. add(v2)). add(v3)。
代碼中還有一個(gè)在Python類定義經(jīng)常使用的 __str__ 函數(shù),當(dāng)使用 str() 時(shí)會(huì)被調(diào)用。print函數(shù)對(duì)傳入的參數(shù)都調(diào)用了str()將其轉(zhuǎn)換成易讀的字符串形式,便于打印輸出,因而會(huì)調(diào)用類定義的__str__函數(shù)打出自定義的字符串。
代碼中還有一個(gè)特殊的 __call__ 函數(shù),該函數(shù)在將對(duì)象采用函數(shù)調(diào)用方式使用時(shí)被調(diào)用, 例如v1()相當(dāng)于v1. call ()。
以上就是魔法函數(shù)的基本使用方法。常見的魔法函數(shù)我們可以使用 dir() 函數(shù)來查看。
輸出結(jié)果為:
上述結(jié)果中形式為‘__函數(shù)名__’的函數(shù)為魔法函數(shù),注意有些對(duì)象也是這種形式,例如__class__, __module__等, 這些不是魔法函數(shù)。具體的魔法函數(shù)說明可以參考Python官方說明文檔。
以上代碼在Python3.6運(yùn)行通過.