Python中的@staticmethod和@classmethod的區別

一直搞不明白,類方法和靜態方法的區別,特意研究了一下,跟大家分享一下。

為了方便大家了解兩者的差別,以下的示例程式碼將有助於發現其中的差別:

class A(object):
    def foo(self, x):
        print "executing foo(%s, %s)" % (self, x)

    @classmethod
    def class_foo(cls, x):
        print "executing class_foo(%s, %s)" % (cls, x)

    @staticmethod
    def static_foo(x):
        print "executing static_foo(%s)" % x    

a = A()

以下是對象實例調用方法的常用方法,對象實例a作為第一個參數隱式傳遞。

a.foo(1)
# executing foo(<__main__.A object at 0xb7dbef0c>,1)

使用classmethods時,對象實例的類作為第一個參數而不是隱式傳遞self。

a.class_foo(1)
# executing class_foo(<class '__main__.A'>,1)

您也可以class_foo使用該類進行調用。

實際上,如果您將某些東西定義為類方法,則可能是因為您打算從類而不是從類實例調用它。

A.foo(1)本來會引發TypeError,但A.class_foo(1)效果很好:

A.class_foo(1)
# executing class_foo(<class '__main__.A'>,1)

人們發現類方法的一種用途是創建可繼承的替代構造函數。

使用staticmethods時,self(對象實例)和 cls(類)都不會隱式傳遞為第一個參數。它們的行為類似於普通函數,只是您可以從實例或類中調用它們:

a.static_foo(1)
# executing static_foo(1)

A.static_foo('hi')
# executing static_foo(hi)

特別注意此句:

靜態方法用於對與類之間具有某種邏輯聯繫的函數分組。

foo只是一個函數,但是當您調用a.foo它時,不僅獲得該函數,還會獲得該函數的「部分應用」版本,該對象實例a綁定為該函數的第一個參數。foo期望有2個參數,而a.foo只期望有1個參數。

a勢必到foo。這就是下面的術語「綁定」的含義:

print(a.foo)
# <bound method A.foo of <__main__.A object at 0xb7d52f0c>>

與a.class_foo,a不綁定class_foo,而是與類A綁定class_foo。

print(a.class_foo)
# <bound method type.class_foo of <class '__main__.A'>>

在這裡,使用靜態方法,即使它是一種方法,也a.static_foo只是返回一個沒有綁定參數的良好的’ole函數。static_foo期望有1個參數,也 a.static_foo期望有1個參數。

print(a.static_foo)
# <function static_foo at 0xb7d479cc>

當然,當您static_foo使用類進行調用時,也會發生同樣的事情A。

print(A.static_foo)
# <function static_foo at 0xb7d479cc>

總結一下彼此的調用區別:

e9c9bb5d9285c7eae68b281b8ab094a0.png

本文首發於BigYoung小站