国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > Python > 正文

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

2019-11-08 01:10:57
字體:
來源:轉載
供稿:網友

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

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

實例來說明

Python
1234567classTest:    defprt(self):        print(self)        print(self.__class__) t =Test()t.prt()

執行結果如下

Python
12<__main__.Testobject at0x000000000284E080><class'__main__.Test'>

從上面的例子中可以很明顯的看出,self代表的是類的實例。而self.class則指向類。

self不必非寫成self

有很多童鞋是先學習別的語言然后學習Python的,所以總覺得self怪怪的,想寫成this,可以嗎?

當然可以,還是把上面的代碼改寫一下。

Python
1234567classTest:    defprt(this):        print(this)        print(this.__class__) t =Test()t.prt()

改成this后,運行結果完全一樣。

當然,最好還是尊重約定俗成的習慣,使用self。

self可以不寫嗎

在Python的解釋器內部,當我們調用t.prt()時,實際上Python解釋成Test.prt(t),也就是說把self替換成類的實例。

有興趣的童鞋可以把上面的t.prt()一行改寫一下,運行后的實際結果完全相同。

實際上已經部分說明了self在定義時不可以省略,如果非要試一下,那么請看下面:

Python
123456classTest:    defprt():        print(self) t= Test()t.prt()

運行時提醒錯誤如下:prt在定義時沒有參數,但是我們運行時強行傳了一個參數。

由于上面解釋過了t.prt()等同于Test.prt(t),所以程序提醒我們多傳了一個參數t。

Python
1234Traceback(most recentcall last):  File"h.py",line 6,in <module>    t.prt()TypeError:prt()takes 0positional arguments but 1was given

當然,如果我們的定義和調用時均不傳類實例是可以的,這就是類方法。

Python
1234classTest:    defprt():        print(__class__)Test.prt()

運行結果如下

Python
1<class'__main__.Test'>

在繼承時,傳入的是哪個實例,就是那個傳入的實例,而不是指定義了self的類的實例。

先看代碼

Python
123456789101112classParent:    defpprt(self):        print(self) classChild(Parent):    defcprt(self):        print(self)c =Child()c.cprt()c.pprt()p= Parent()p.pprt()

運行結果如下

Python
123<__main__.Childobject at0x0000000002A47080><__main__.Childobject at0x0000000002A47080><__main__.Parentobject at0x0000000002A47240>

解釋:

運行c.cprt()時應該沒有理解問題,指的是Child類的實例。

但是在運行c.pprt()時,等同于Child.pprt(c),所以self指的依然是Child類的實例,由于self中沒有定義pprt()方法,所以沿著繼承樹往上找,發現在父類Parent中定義了pprt()方法,所以就會成功調用。

在描述符類中,self指的是描述符類的實例

不太容易理解,先看實例:

Python
1234567891011classDesc:    def__get__(self,ins,cls):        print('self in Desc: %s '% self)        print(self,ins,cls)classTest:    x= Desc()    defprt(self):        print('self in Test: %s'% self)t= Test()t.prt()t.x

運行結果如下:

Python
123selfin Test:<__main__.Testobject at0x0000000002A570B8>self inDesc:<__main__.Descobject at0x000000000283E208><__main__.Descobject at0x000000000283E208><__main__.Testobject at0x0000000002A570B8><class'__main__.Test'>

大部分童鞋開始有疑問了,為什么在Desc類中定義的self不是應該是調用它的實例t嗎?怎么變成了Desc類的實例了呢?

注意:此處需要睜大眼睛看清楚了,這里調用的是t.x,也就是說是Test類的實例t的屬性x,由于實例t中并沒有定義屬性x,所以找到了類屬性x,而該屬性是描述符屬性,為Desc類的實例而已,所以此處并沒有頂用Test的任何方法。

那么我們如果直接通過類來調用屬性x也可以得到相同的結果。

下面是把t.x改為Test.x運行的結果。

Python
123selfin Test:<__main__.Testobject at0x00000000022570B8>self inDesc:<__main__.Descobject at0x000000000223E208><__main__.Descobject at0x000000000223E208>None <class'__main__.Test'>

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

 

總結

self在定義時需要定義,但是在調用時會自動傳入。self的名字并不是規定死的,但是最好還是按照約定是用selfself總是指調用時的類的實例。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 托克托县| 万宁市| 华亭县| 沙雅县| 宜城市| 盱眙县| 遂平县| 谢通门县| 大理市| 铜鼓县| 许昌市| 伊宁县| 镇赉县| 房山区| 江北区| 饶河县| 淮安市| 顺平县| 呼和浩特市| 淳安县| 大同市| 兴宁市| 建阳市| 乌鲁木齐县| 庄浪县| 五家渠市| 四会市| 囊谦县| 鄂托克前旗| 仪陇县| 西安市| 谢通门县| 墨脱县| 永靖县| 永宁县| 芦山县| 宝坻区| 漳浦县| 丰顺县| 太谷县| 乌什县|