Python的靜態方法和類別成員方法都可以被類別或實例訪問,兩者概念不容易理清,但還是有區別的:
1)靜態方法無需傳入self參數,類別成員方法需傳入代表本類別的cls參數;(參數cls代表本類)
2)從第1條,靜態方法是無法存取實例變數的,而類別成員方法也同樣無法存取實例變量,但可以存取類別變數;
3)靜態方法有點像是函數工具庫的作用,而類別成員方法則更接近類似Java物件導向概念中的靜態方法。
實作靜態方法和類別方法的兩種方式
一、在Python 2.3及之前,用staticmethod和classmethod類型物件包裝實作
MyClass: val1 = 'Value 1'
def __init__(self): self.val2 = 'Value 2'
def staticmd(): 121 smd = staticmethod(staticmd)
def classmd(cls):
print '類別方法,類別:' + str(cls) + ',val1:' + cls.val1 + ',無法存取val2的值'
>>> mc = MyClass()
>>> mc.smd()
>>> mc.cmd()
>>> MyClass.smd()
>>二、在Python 2.4及以後,用裝飾器(decorators)實現
裝飾器使用@操作符,例子如下:
class MyClass:
val1 = 'Value 1'
: = 'Value 2'
@staticmethod
print '靜態方法,
print '類別方法,類別:' + str( cls) + ',val1:' + cls.val1 + ',無法存取val2的值'
不管是以上兩種方式中的哪一種,執行情況都是一樣的,以方式二執行結果為例分析如下:
執行:
>>> mc = MyClass() # 實例化
>>> mc.staticmd() # 實例化
>靜態方法,無法存取val1和val2
>>> mc.classmd() # 實例呼叫類別方法,注意,這裡存取的是類別MyClass的變數val1的值,不是實例化後mc的實例變數val1,這裡容易混淆,往下看就會明白。 val2一直是實例變量,所以無法存取
>>>
類別方法,類別:__main__.MyClass,val1:Value 1,無法存取val2的值
>>> MyClass.staticmd() 靜態方法,結果同上面的實例調用,無論是類別變數或實例變數都無法存取
>>> 1和val2
>>> 類別方法,結果同上面的實例呼叫
>>>
類別方法,類別:__main__.MyClass,val1:Value 1,無法存取val2的值
'>>> # 改變實例變數val1的值
類別方法,類別:__main__.MyClass,val1:Value 1,無法存取val2的值
>>>
類別方法,類別:__main__.MyClass,val1:Class Value changed,無法存取val2的值
>>> MyClass.classmd() # 類別直接呼叫類別方法,結果同上面的實例呼叫
> > 類別方法,類別:__main__.MyClass,val1:Class Value changed,無法存取val2的值 結論如果上述執行過程就好太複雜,記住以下兩點:無法存取類別屬性、實例屬性,相當於一個相對獨立的方法,跟類別其實沒什麼關係,換個角度來講,其實就是放在一個類別的作用域裡的函數而已。 類別成員方法:可以存取類別屬性,無法存取實例屬性。上述的變數val1,在類別裡是類別變量,在實例中又是實例變量,所以容易混淆。