1. 程式人生 > >終於明白了Python中self的含義

終於明白了Python中self的含義

剛開始學習Python的類寫法的時候覺得很是麻煩,為什麼定義時需要而呼叫時又不需要,為什麼不能內部簡化從而減少我們敲擊鍵盤的次數? 你看完這篇文章後就會明白所有的疑問。

self代表類的例項,而非類。 例項來說明

class Test:     def prt(self):         print(self)         print(self.__class__)   t = Test() t.prt() 執行結果如下

<__main__.Test object at 0x000000000284E080> <class '__main__.Test'> 從上面的例子中可以很明顯的看出,self代表的是類的例項。而self.__class__則指向類。

self不必非寫成self 有很多童鞋是先學習別的語言然後學習Python的,所以總覺得self怪怪的,想寫成this,可以嗎? 當然可以,還是把上面的程式碼改寫一下。

class Test:     def prt(this):         print(this)         print(this.__class__)   t = Test() t.prt() 改成this後,執行結果完全一樣。 當然,最好還是尊重約定俗成的習慣,使用self。

self可以不寫嗎 在Python的直譯器內部,當我們呼叫t.prt()時,實際上Python解釋成Test.prt(t),也就是說把self替換成類的例項。 有興趣的童鞋可以把上面的t.prt()一行改寫一下,執行後的實際結果完全相同。 實際上已經部分說明了self在定義時不可以省略,如果非要試一下,那麼請看下面:

class Test:     def prt():         print(self)   t = Test() t.prt() 執行時提醒錯誤如下:prt在定義時沒有引數,但是我們執行時強行傳了一個引數。 由於上面解釋過了t.prt()等同於Test.prt(t),所以程式提醒我們多傳了一個引數t。

Traceback (most recent call last):   File "h.py", line 6, in <module>     t.prt() TypeError: prt() takes 0 positional arguments but 1 was given 當然,如果我們的定義和呼叫時均不傳類例項是可以的,這就是類方法。

class Test:     def prt():         print(__class__) Test.prt() 執行結果如下

<class '__main__.Test'>  在繼承時,傳入的是哪個例項,就是那個傳入的例項,而不是指定義了self的類的例項。 先看程式碼

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() 執行結果如下

<__main__.Child object at 0x0000000002A47080> <__main__.Child object at 0x0000000002A47080> <__main__.Parent object at 0x0000000002A47240> 解釋: 執行c.cprt()時應該沒有理解問題,指的是Child類的例項。 但是在執行c.pprt()時,等同於Child.pprt(c),所以self指的依然是Child類的例項,由於self中沒有定義pprt()方法,所以沿著繼承樹往上找,發現在父類Parent中定義了pprt()方法,所以就會成功呼叫。

在描述符類中,self指的是描述符類的例項 不太容易理解,先看例項:

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 執行結果如下:

self in Test: <__main__.Test object at 0x0000000002A570B8> self in Desc: <__main__.Desc object at 0x000000000283E208> <__main__.Desc object at 0x000000000283E208> <__main__.Test object at 0x0000000002A570B8> <class '__main__.Test'> 大部分童鞋開始有疑問了,為什麼在Desc類中定義的self不是應該是呼叫它的例項t嗎?怎麼變成了Desc類的例項了呢? 注意:此處需要睜大眼睛看清楚了,這裡呼叫的是t.x,也就是說是Test類的例項t的屬性x,由於例項t中並沒有定義屬性x,所以找到了類屬性x,而該屬性是描述符屬性,為Desc類的例項而已,所以此處並沒有頂用Test的任何方法。 那麼我們如果直接通過類來呼叫屬性x也可以得到相同的結果。 下面是把t.x改為Test.x執行的結果。

self in Test: <__main__.Test object at 0x00000000022570B8> self in Desc: <__main__.Desc object at 0x000000000223E208> <__main__.Desc object at 0x000000000223E208> None <class '__main__.Test'> 題外話:由於在很多時候描述符類中仍然需要知道呼叫該描述符的例項是誰,所以在描述符類中存在第二個引數ins,用來表示呼叫它的類例項,所以t.x時可以看到第三行中的執行結果中第二項為<main.Test object at 0x0000000002A570B8>。而採用Test.x進行呼叫時,由於沒有例項,所以返回None。

總結 self在定義時需要定義,但是在呼叫時會自動傳入。 self的名字並不是規定死的,但是最好還是按照約定是用self self總是指呼叫時的類的例項。 以上所有程式碼在Python3.4中均測試通過。 ---------------------  作者:Bonnie-幫你  來源:CSDN  原文:https://blog.csdn.net/daocaoren1543169565/article/details/80626035  版權宣告:本文為博主原創文章,轉載請附上博文連結!