在學(xué)習(xí)python代碼時(shí),看到有的類(lèi)的方法中第一參數(shù)是cls,有的是self,經(jīng)過(guò)了解得知,python并沒(méi)有對(duì)類(lèi)中方法的第一個(gè)參數(shù)名字做限制,可以是self,也可以是cls,不過(guò)根據(jù)人們的慣用用法,self一般是在實(shí)例方法中使用,而cls則一般在類(lèi)方法中使用,在靜態(tài)方法中則不需要使用一個(gè)默認(rèn)參數(shù)。在下面的代碼中,InstanceMethod類(lèi)的方法中,第一個(gè)參數(shù)是默認(rèn)的self,在這里可以把self換成任何名字來(lái)表示,不會(huì)有任何影響。在類(lèi)調(diào)用的時(shí)候,需要滿足參數(shù)的個(gè)數(shù)要求(參數(shù)中含有*args是例外),例如13行中,類(lèi)調(diào)用沒(méi)有參數(shù)的時(shí)候,會(huì)提示錯(cuò)誤。同樣,實(shí)例方法的參數(shù)個(gè)數(shù)也應(yīng)該滿足要求,例如16行中也會(huì)報(bào)錯(cuò)。實(shí)例方法的一個(gè)主要特點(diǎn)就是需要綁定到一個(gè)對(duì)象上,python解析器會(huì)自動(dòng)把實(shí)例自身傳遞給方法,如14行所示,而直接使用InstanceMethod.f1()調(diào)用方法是不行的。
創(chuàng)新互聯(lián)堅(jiān)持“要么做到,要么別承諾”的工作理念,服務(wù)領(lǐng)域包括:網(wǎng)站設(shè)計(jì)、做網(wǎng)站、企業(yè)官網(wǎng)、英文網(wǎng)站、手機(jī)端網(wǎng)站、網(wǎng)站推廣等服務(wù),滿足客戶于互聯(lián)網(wǎng)時(shí)代的濟(jì)水街道網(wǎng)站設(shè)計(jì)、移動(dòng)媒體設(shè)計(jì)的需求,幫助企業(yè)找到有效的互聯(lián)網(wǎng)解決方案。努力成為您成熟可靠的網(wǎng)絡(luò)建設(shè)合作伙伴!
12345678910111213
class InstanceMethod(object): def __init__(self, a): self.a = a def f1(self): print 'This is {0}.'.format(self) def f2(self, a): print 'Value:{0}'.format(a)if __name__ == '__main__': # im = InstanceMethod() im = InstanceMethod('233') im.f1() # im.f2() im.f2(233)
靜態(tài)方法和類(lèi)方法都需要使用修飾器,分別使用的是staticmethod和classmethod。靜態(tài)方法是和類(lèi)沒(méi)有關(guān)系的,我覺(jué)得就是包裝在類(lèi)中的一般方法,如下例子中,調(diào)用靜態(tài)方法使用實(shí)例和不使用實(shí)例都是可以的。類(lèi)方法中,默認(rèn)的第一個(gè)參數(shù)使用的是cls,類(lèi)方法也可以不需要實(shí)例而直接使用類(lèi)調(diào)用。對(duì)于這三種不同的方法,使用方法如下例所示。那么問(wèn)題來(lái)了,既然有了實(shí)例方法,類(lèi)方法和靜態(tài)方法與之相比又有什么好處呢?
在類(lèi)方法中,不管是使用實(shí)例還是類(lèi)調(diào)用方法,都會(huì)把類(lèi)作為第一個(gè)參數(shù)傳遞進(jìn)來(lái),這個(gè)參數(shù)就是類(lèi)本身。如果繼承了這個(gè)使用類(lèi)方法的類(lèi),該類(lèi)的所有子類(lèi)都會(huì)擁有了這個(gè)方法,并且這個(gè)方法會(huì)自動(dòng)指向子類(lèi)本身,這個(gè)特性在工廠函數(shù)中是非常有用的。靜態(tài)方法是和類(lèi)與實(shí)例都沒(méi)有關(guān)系的,完全可以使用一般方法代替,但是使用靜態(tài)方法可以更好的組織代碼,防止代碼變大后變得比較混亂。類(lèi)方法是可以替代靜態(tài)方法的。靜態(tài)方法不能在繼承中修改。
123456789101112131415161718
class test(object): def instance_method(self): print 'This is {0}'.format(self) @staticmethod def static_method(): print 'This is static method.' @classmethod def class_method(cls): print 'This is {0}'.format(cls)if __name__ == '__main__': a = test() a.instance_method() a.static_method() a.class_method() print '----------------------------------------' # test.instance_method() test.static_method() test.class_method()
python的類(lèi)里用@staticmethod的是靜態(tài)方法,@classmethod的是類(lèi)方法,如下
class?Person(object):
person_list?=?[]
def?__init__(self,?name,?age):
self.name?=?name
self.age?=?age
self.person_list.append(self)
@classmethod
def?one_year_later(cls):
for?p?in?cls.person_list:
p.age?+=?1
@staticmethod
def?born_one_boby(name):
return?Person(name,?0)
def?__repr__(self):
return?'Person?name:%s,?age:%s'?%?(self.name,?self.age)
if?__name__?==?'__main__':
petter?=??Person('Petter',23)
merry?=?Person('Merry',21)
print(petter)?#?Person?name:Petter,?age:23
print(merry)??#??Person?name:Merry,?age:21
Person.one_year_later()
print(petter)??#??Person?name:Petter,?age:24
print(merry)??#??Person?name:Merry,?age:22
baby?=?merry.born_one_boby('Tom')
print(Person.person_list)?#??[Person?name:Petter,?age:24,?Person?name:Merry,?age:22,?Person?name:Tom,?age:0]
Python使用靜態(tài)方法類(lèi)似函數(shù)工具使用,一般盡量少用靜態(tài)方法。
Python的靜態(tài)方法和類(lèi)成員方法都可以被類(lèi)或?qū)嵗L問(wèn),兩者概念不容易理清,但還是有區(qū)別的:
1)靜態(tài)方法無(wú)需傳入self參數(shù),類(lèi)成員方法需傳入代表本類(lèi)的cls參數(shù);
2)從第1條,靜態(tài)方法是無(wú)法訪問(wèn)實(shí)例變量的,而類(lèi)成員方法也同樣無(wú)法訪問(wèn)實(shí)例變量,但可以訪問(wèn)類(lèi)變量;
3)靜態(tài)方法有點(diǎn)像函數(shù)工具庫(kù)的作用,而類(lèi)成員方法則更接近類(lèi)似Java面向?qū)ο蟾拍钪械撵o態(tài)方法。
面相對(duì)象程序設(shè)計(jì)中,類(lèi)方法和靜態(tài)方法是經(jīng)常用到的兩個(gè)術(shù)語(yǔ)。
邏輯上講:類(lèi)方法是只能由類(lèi)名調(diào)用;靜態(tài)方法可以由類(lèi)名或?qū)ο竺M(jìn)行調(diào)用。
在C++中,靜態(tài)方法與類(lèi)方法邏輯上是等價(jià)的,只有一個(gè)概念,不會(huì)混淆。
而在python中,方法分為三類(lèi)實(shí)例方法、類(lèi)方法、靜態(tài)方法。代碼如下:
class Test(object):
def InstanceFun(self):
print("InstanceFun");
print(self);
@classmethod
def ClassFun(cls):
print("ClassFun");
print(cls);
@staticmethod
def StaticFun():
print("StaticFun");
t = Test();
t.InstanceFun();# 輸出InstanceFun,打印對(duì)象內(nèi)存地址“__main__.Test object at 0x0293DCF0”
Test.ClassFun(); # 輸出ClassFun,打印類(lèi)位置 class '__main__.Test'
Test.StaticFun(); # 輸出StaticFun
t.StaticFun(); # 輸出StaticFun
t.ClassFun(); # 輸出ClassFun,打印類(lèi)位置 class '__main__.Test'
Test.InstanceFun(); # 錯(cuò)誤,TypeError: unbound method instanceFun() must be called with Test instance as first argument
Test.InstanceFun(t); # 輸出InstanceFun,打印對(duì)象內(nèi)存地址“__main__.Test object at 0x0293DCF0”
t.ClassFun(Test); # 錯(cuò)誤 classFun() takes exactly 1 argument (2 given)
可以看到,在python中,兩種方法的主要區(qū)別在于參數(shù)。實(shí)例方法隱含的參數(shù)為類(lèi)實(shí)例self,而類(lèi)方法隱含的參數(shù)為類(lèi)本身cls。
靜態(tài)方法無(wú)隱含參數(shù),主要為了類(lèi)實(shí)例也可以直接調(diào)用靜態(tài)方法。
所以邏輯上類(lèi)方法應(yīng)當(dāng)只被類(lèi)調(diào)用,實(shí)例方法實(shí)例調(diào)用,靜態(tài)方法兩者都能調(diào)用。主要區(qū)別在于參數(shù)傳遞上的區(qū)別,實(shí)例方法悄悄傳遞的是self引用作為參數(shù),而類(lèi)方法悄悄傳遞的是cls引用作為參數(shù)。
python實(shí)現(xiàn)了一定的靈活性使得類(lèi)方法和靜態(tài)方法,都能夠被實(shí)例和類(lèi)二者調(diào)用
@staticmethod 靜態(tài)方法只是名義上歸屬類(lèi)管理,但是 不能使用類(lèi)變量和實(shí)例變量,是類(lèi)的工具包
放在函數(shù)前(該函數(shù)不傳入self或者cls),所以不能訪問(wèn)類(lèi)屬性和實(shí)例屬性