python中系列的含义_一篇文章让你彻底搞清楚Python中self的含义
剛開(kāi)始學(xué)習(xí)Python的類寫(xiě)法的時(shí)候覺(jué)得很是麻煩,為什么定義時(shí)需要而調(diào)用時(shí)又不需要,為什么不能內(nèi)部簡(jiǎn)化從而減少我們敲擊鍵盤(pán)的次數(shù)?
你看完這篇文章后就會(huì)明白所有的疑問(wèn)。
self代表類的實(shí)例,而非類。
實(shí)例來(lái)說(shuō)明
class Test:
def prt(self):
print(self)
print(self.__class__)
t = Test()
t.prt()
執(zhí)行結(jié)果如下
從上面的例子中可以很明顯的看出,self代表的是類的實(shí)例。而self.__class__則指向類。
self不必非寫(xiě)成self
有很多童鞋是先學(xué)習(xí)別的語(yǔ)言然后學(xué)習(xí)Python的,所以總覺(jué)得self怪怪的,想寫(xiě)成this,可以嗎?
當(dāng)然可以,還是把上面的代碼改寫(xiě)一下。
class Test:
def prt(this):
print(this)
print(this.__class__)
t = Test()
t.prt()
改成this后,運(yùn)行結(jié)果完全一樣。
當(dāng)然,最好還是尊重約定俗成的習(xí)慣,使用self。
self可以不寫(xiě)嗎
在Python的解釋器內(nèi)部,當(dāng)我們調(diào)用t.prt()時(shí),實(shí)際上Python解釋成Test.prt(t),也就是說(shuō)把self替換成類的實(shí)例。
有興趣的童鞋可以把上面的t.prt()一行改寫(xiě)一下,運(yùn)行后的實(shí)際結(jié)果完全相同。
實(shí)際上已經(jīng)部分說(shuō)明了self在定義時(shí)不可以省略,如果非要試一下,那么請(qǐng)看下面:
class Test:
def prt():
print(self)
t = Test()
t.prt()
運(yùn)行時(shí)提醒錯(cuò)誤如下:prt在定義時(shí)沒(méi)有參數(shù),但是我們運(yùn)行時(shí)強(qiáng)行傳了一個(gè)參數(shù)。
由于上面解釋過(guò)了t.prt()等同于Test.prt(t),所以程序提醒我們多傳了一個(gè)參數(shù)t。
Traceback (most recent call last):
File "h.py", line 6, in
t.prt()
TypeError: prt() takes 0 positional arguments but 1 was given
當(dāng)然,如果我們的定義和調(diào)用時(shí)均不傳類實(shí)例是可以的,這就是類方法。
class Test:
def prt():
print(__class__)
Test.prt()
運(yùn)行結(jié)果如下
在繼承時(shí),傳入的是哪個(gè)實(shí)例,就是那個(gè)傳入的實(shí)例,而不是指定義了self的類的實(shí)例。
先看代碼
class Parent:
def pprt(self):
print(self)
class Child(Parent):
def cprt(self):
print(self)
c = Child()
c.cprt()
c.pprt()
p = Parent()
p.pprt()
運(yùn)行結(jié)果如下
解釋:
運(yùn)行c.cprt()時(shí)應(yīng)該沒(méi)有理解問(wèn)題,指的是Child類的實(shí)例。
但是在運(yùn)行c.pprt()時(shí),等同于Child.pprt(c),所以self指的依然是Child類的實(shí)例,由于self中沒(méi)有定義pprt()方法,所以沿著繼承樹(shù)往上找,發(fā)現(xiàn)在父類Parent中定義了pprt()方法,所以就會(huì)成功調(diào)用。
在描述符類中,self指的是描述符類的實(shí)例
不太容易理解,先看實(shí)例:
class Desc:
def __get__(self, ins, cls):
print('self in Desc: %s ' % self )
print(self, ins, cls)
class Test:
x = Desc()
def prt(self):
print('self in Test: %s' % self)
t = Test()
t.prt()
t.x
運(yùn)行結(jié)果如下:
self in Test: <__main__.test object at>
self in Desc: <__main__.desc object at>
大部分童鞋開(kāi)始有疑問(wèn)了,為什么在Desc類中定義的self不是應(yīng)該是調(diào)用它的實(shí)例t嗎?怎么變成了Desc類的實(shí)例了呢?
注意:此處需要睜大眼睛看清楚了,這里調(diào)用的是t.x,也就是說(shuō)是Test類的實(shí)例t的屬性x,由于實(shí)例t中并沒(méi)有定義屬性x,所以找到了類屬性x,而該屬性是描述符屬性,為Desc類的實(shí)例而已,所以此處并沒(méi)有頂用Test的任何方法。
那么我們?nèi)绻苯油ㄟ^(guò)類來(lái)調(diào)用屬性x也可以得到相同的結(jié)果。
下面是把t.x改為T(mén)est.x運(yùn)行的結(jié)果。
self in Test: <__main__.test object at>
self in Desc: <__main__.desc object at>
<__main__.desc object at> None
題外話:由于在很多時(shí)候描述符類中仍然需要知道調(diào)用該描述符的實(shí)例是誰(shuí),所以在描述符類中存在第二個(gè)參數(shù)ins,用來(lái)表示調(diào)用它的類實(shí)例,所以t.x時(shí)可以看到第三行中的運(yùn)行結(jié)果中第二項(xiàng)為。而采用Test.x進(jìn)行調(diào)用時(shí),由于沒(méi)有實(shí)例,所以返回None。
總結(jié)
self在定義時(shí)需要定義,但是在調(diào)用時(shí)會(huì)自動(dòng)傳入。
self的名字并不是規(guī)定死的,但是最好還是按照約定是用self
self總是指調(diào)用時(shí)的類的實(shí)例。
以上所有代碼在Python3.4中均測(cè)試通過(guò)。
總結(jié)
以上是生活随笔為你收集整理的python中系列的含义_一篇文章让你彻底搞清楚Python中self的含义的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: python函数式编程读取数据时出现错误
- 下一篇: python定义字典列表_[Python