python類的實例方法立叛、靜態(tài)方法和類方法區(qū)別及其應(yīng)用場景
一、先看語法贡茅,python 類語法中有三種方法秘蛇,實例方法,靜態(tài)方法顶考,類方法赁还。
ps.python中self,cls的區(qū)別
普通實例方法村怪,第一個參數(shù)需要是self秽浇,它表示一個具體的實例本身。
如果用了staticmethod甚负,那么就可以無視這個self柬焕,而將這個方法當成一個普通的函數(shù)使用。
而對于classmethod梭域,它的第一個參數(shù)不是self斑举,是cls,它表示這個類本身病涨。
# coding:utf-8
class Foo(object):
"""類三種方法語法形式"""
def instance_method(self):
print("是類{}的實例方法富玷,只能被實例對象調(diào)用".format(Foo))
@staticmethod
def static_method():
print("是靜態(tài)方法")
@classmethod
def class_method(cls):
print("是類方法")
foo = Foo()
foo.instance_method()
foo.static_method()
foo.class_method()
print('----------------')
Foo.static_method()
Foo.class_method()
運行結(jié)果如下
是類<class '__main__.Foo'>的實例方法,只能被實例對象調(diào)用
是靜態(tài)方法
是類方法
----------------
是靜態(tài)方法
是類方法
說明:
實例方法只能被實例對象調(diào)用既穆,靜態(tài)方法(由@staticmethod裝飾的方法)赎懦、類方法(由@classmethod裝飾的方法),可以被類或類的實例對象調(diào)用幻工。
實例方法励两,第一個參數(shù)必須要默認傳實例對象,一般習(xí)慣用self囊颅。
靜態(tài)方法当悔,參數(shù)沒有要求。
類方法踢代,第一個參數(shù)必須要默認傳類盲憎,一般習(xí)慣用cls。
二胳挎、靜態(tài)方法饼疙、類方法使用區(qū)別或者說使用場景
1、類方法用在模擬java定義多個構(gòu)造函數(shù)的情況串远。
由于python類中只能有一個初始化方法宏多,不能按照不同的情況初始化類儿惫。
參考django https://docs.djangoproject.com/en/1.9/ref/models/instances/ 請看下面的代碼澡罚。
# coding:utf-8
class Book(object):
def __init__(self, title):
self.title = title
@classmethod
def class_method_create(cls, title):
book = cls(title=title)
return book
@staticmethod
def static_method_create(title):
book= Book(title)
return book
book1 = Book("use instance_method_create book instance")
book2 = Book.class_method_create("use class_method_create book instance")
book3 = Book.static_method_create("use static_method_create book instance")
print(book1.title)
print(book2.title)
print(book3.title)
結(jié)果:
use instance_method_create book instance
use class_method_create book instance
use static_method_create book instance
Process finished with exit code 0
特別說明伸但,靜態(tài)方法也可以實現(xiàn)上面功能,當靜態(tài)方法每次都要寫上類的名字留搔,不方便更胖。
2、類中靜態(tài)方法方法調(diào)用靜態(tài)方法和類方法調(diào)用靜態(tài)方法例子隔显。 下面的代碼却妨,靜態(tài)方法調(diào)用另一個靜態(tài)方法,如果改用類方法調(diào)用靜態(tài)方法括眠,可以讓cls代替類彪标,
讓代碼看起來精簡一些。也防止類名修改了掷豺,不用在類定義中修改原來的類名捞烟。
# coding:utf-8class Foo(object): X = 1 Y = 2 @staticmethod def averag(*mixes): return sum(mixes) / len(mixes) @staticmethod def static_method(): # 在靜態(tài)方法中調(diào)用靜態(tài)方法 print "在靜態(tài)方法中調(diào)用靜態(tài)方法" return Foo.averag(Foo.X, Foo.Y) @classmethod def class_method(cls): # 在類方法中使用靜態(tài)方法 print "在類方法中使用靜態(tài)方法" return cls.averag(cls.X, cls.Y) foo = Foo()print(foo.static_method())print(foo.class_method())
結(jié)果:
在靜態(tài)方法中調(diào)用靜態(tài)方法
5
在類方法中使用靜態(tài)方法
5
**3、繼承類中的區(qū)別 **
從下面代碼可以看出当船,如果子類繼承父類的方法题画,子類覆蓋了父類的靜態(tài)方法,
子類的實例繼承了父類的static_method靜態(tài)方法德频,調(diào)用該方法苍息,還是調(diào)用的父類的方法和類屬性。
子類的實例繼承了父類的class_method類方法壹置,調(diào)用該方法竞思,調(diào)用的是子類的方法和子類的類屬性。
# coding:utf-8
class Foo(object):
X = 1
Y = 14
@staticmethod
def averag(*mixes):
# "父類中的靜態(tài)方法"
return sum(mixes) / len(mixes)
@staticmethod
def static_method():
# "父類中的靜態(tài)方法"
print "父類中的靜態(tài)方法"
return Foo.averag(Foo.X, Foo.Y)
@classmethod
def class_method(cls):
# 父類中的類方法
print "父類中的類方法"
return cls.averag(cls.X, cls.Y)
class Son(Foo):
X = 3
Y = 5
@staticmethod
def averag(*mixes):
# "子類中重載了父類的靜態(tài)方法"
print "子類中重載了父類的靜態(tài)方法"
return sum(mixes) / 3
p = Son()
print "result of p.averag(1,5)"
print (p.averag(1,5))
print "result of p.static_method()"
print(p.static_method())
print "result of p.class_method()"
print(p.class_method())
結(jié)果如下:
result of p.averag(1,5)
子類中重載了父類的靜態(tài)方法
2
result of p.static_method()
父類中的靜態(tài)方法
7
result of p.class_method()
父類中的類方法
子類中重載了父類的靜態(tài)方法
2
Process finished with exit code 0