乡下人产国偷v产偷v自拍,国产午夜片在线观看,婷婷成人亚洲综合国产麻豆,久久综合给合久久狠狠狠9

  • <output id="e9wm2"></output>
    <s id="e9wm2"><nobr id="e9wm2"><ins id="e9wm2"></ins></nobr></s>

    • 分享

      一篇文章讓你徹底搞清楚Python中self的含義

       刮骨劍 2019-07-11

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

      self代表類的實例,而非類。

      實例來說明

      class Test:
          def prt(self):
              print(self)
              print(self.__class__)
      
      t = Test()
      t.prt()

      執(zhí)行結果如下

      <__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()一行改寫一下,運行后的實際結果完全相同。
      實際上已經(jīng)部分說明了self在定義時不可以省略,如果非要試一下,那么請看下面:

      class Test:
          def prt():
              print(self)
      
      t = Test()
      t.prt()

      運行時提醒錯誤如下:prt在定義時沒有參數(shù),但是我們運行時強行傳了一個參數(shù)。
      由于上面解釋過了t.prt()等同于Test.prt(t),所以程序提醒我們多傳了一個參數(shù)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()方法,所以沿著繼承樹往上找,發(fā)現(xiàn)在父類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'>

      題外話:由于在很多時候描述符類中仍然需要知道調用該描述符的實例是誰,所以在描述符類中存在第二個參數(shù)ins,用來表示調用它的類實例,所以t.x時可以看到第三行中的運行結果中第二項為<main.Test object at 0x0000000002A570B8>。而采用Test.x進行調用時,由于沒有實例,所以返回None。

      總結

      • self在定義時需要定義,但是在調用時會自動傳入。
      • self的名字并不是規(guī)定死的,但是最好還是按照約定是用self
      • self總是指調用時的類的實例。

      以上所有代碼在Python3.4中均測試通過。

        本站是提供個人知識管理的網(wǎng)絡存儲空間,所有內容均由用戶發(fā)布,不代表本站觀點。請注意甄別內容中的聯(lián)系方式、誘導購買等信息,謹防詐騙。如發(fā)現(xiàn)有害或侵權內容,請點擊一鍵舉報。
        轉藏 分享 獻花(0

        0條評論

        發(fā)表

        請遵守用戶 評論公約

        類似文章 更多