python 中的self和cls
python 中的self和cls
一句話描述:self是類(Class)實例化對象,cls就是類(或子類)本身,取決于調(diào)用的是那個類。
@staticmethod 屬于靜態(tài)方法裝飾器,@classmethod屬于類方法裝飾器。我們需要從聲明和使用兩個方面來理解。
詳細(xì)介紹
一般來說,要使用某個類的方法,需要先??實例化一個對象再調(diào)用方法。而使用@staticmethod或@classmethod,就可以不需要實例化,直接類名.方法名()來調(diào)用。這有利于組織代碼,把某些應(yīng)該屬于某個類的函數(shù)給放到那個類里去,同時有利于命名空間的整潔。🤔
首先定義一個類A,類A中有三個函數(shù),foo1為靜態(tài)函數(shù),用@staticmethod裝飾器裝飾,這種方法與類有某種關(guān)系但不需要使用到實例或者類來參與。
class A(object):a = 'a'@staticmethoddef foo1(name):print('hello', name, A.a)def foo2(self, name):print('hello', name, self.a)@classmethoddef foo3(cls, name):print('hello', name, cls.a) class B(A):a = 'b'@staticmethoddef foo1(name):print('hello', name, B.a)def foo2(self, name):print('subclass B')print('hello', name, self.a)@classmethoddef foo3(cls, name):print('hello', name, cls.a)如下兩種方法都可以正常輸出,也就是說
既可以作為類的方法使用,也可以作為類的實例的方法使用。
a = A() b = B() a.foo1("小熊貓") # hello 小熊貓 a A.foo1("小熊貓") # hello 小熊貓 a b.foo1("大熊貓") # subclass B, hello 大熊貓 b B.foo1("大熊貓") # subclass B, hello 大熊貓 bfoo2為正常的函數(shù),是類的實例的函數(shù),調(diào)用方式如下。
將實參實例化對象或者類名稱傳入self對象,取到不同的屬性和方法。
a.foo2("小熊貓") # hello 小熊貓 a A.foo2(a, "小熊貓") # hello 小熊貓 a A.foo2(b, "小熊貓") # hello 小熊貓 b A.foo2(A, "小熊貓") # hello 小熊貓 a A.foo2(B, "小熊貓") # hello 小熊貓 b B.foo2(a, "小熊貓") # subclass B, hello 小熊貓 a b.foo2("小熊貓") # subclass B, hello 小熊貓 bfoo3為類函數(shù),cls作為第一個參數(shù)用來表示類本身. 在類方法中用到,類方法是只與類本身有關(guān)而與實例無關(guān)的方法。如下兩種方法都可以正常輸出。
可以看出,傳入形參cls的值為前面的調(diào)用函數(shù),如果再傳入對象或者類名稱,會報類型錯誤,多傳了一個參數(shù)。
a.foo3("小熊貓") A.foo3("小熊貓") # a.foo3(a, "小熊貓") # TypeError: foo3() takes 2 positional arguments but 3 were given # A.foo3(A, "小熊貓") # TypeError: foo3() takes 2 positional arguments but 3 were given b.foo3("大熊貓") B.foo3("大熊貓")@staticmethod和@classmethod的用法
相同:
- @staticmethod和@classmethod都可以直接類名.方法名()來調(diào)用
區(qū)別:
- 從它們的使用上來看,@staticmethod不需要表示自身對象的self和自身類的cls參數(shù),就跟使用函數(shù)一樣。@classmethod也不需要self參數(shù),但第一個參數(shù)是調(diào)用類方法的類。
- 如果在@staticmethod中要調(diào)用到這個類的一些屬性方法,只能直接類名.屬性名或類名.方法名。
- 而@classmethod因為持有cls參數(shù),可以來調(diào)用類的屬性,類的方法,實例化對象等,避免硬編碼。
重點應(yīng)關(guān)注@staticmethod和@classmethod調(diào)用本類或其他類的函數(shù)和屬性的區(qū)別
例子1:
關(guān)鍵看第二句 subclass B, hello 小熊貓 b,在調(diào)用 B.foo2(B, “小熊貓”) 時,執(zhí)行了B類型下的foo2()方法,該方法無返回值,因此 下句輸出為 hello foo4 None
a = A() a.foo1("小熊貓") # 輸出 hello foo1 小熊貓 a subclass B, hello 小熊貓 b hello foo4 None例子2:
a.foo3("小熊貓") # 輸出 hello foo3 小熊貓 a hello foo2 小熊貓 a hello foo5 None hello foo1 小熊貓 a subclass B, hello 小熊貓 b hello foo4 None hello foo6 None最后一個小例子
class A(object):@classmethoddef acquire(cls, param):if cls is A:raise Exception("只有子類才能調(diào)用!!!")return "類 a %r 的參數(shù)為 %r" % (cls, param)class B(A):passclass C(A):passprint(B.acquire("小熊貓")) print(C.acquire("大熊貓")) print(A.acquire("愛吃竹子")) # 結(jié)果 類 a <class '__main__.B'> 的參數(shù)為 '小熊貓' 類 a <class '__main__.C'> 的參數(shù)為 '大熊貓' Exception: 只有子類才能調(diào)用!!!總結(jié)
以上是生活随笔為你收集整理的python 中的self和cls的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Reg命令使用详解 批处理操作注册表必备
- 下一篇: 9 个 yyds 的 Java 项目,可