全面理解Python中self的用法
剛開始學(xué)習(xí)Python的類寫法的時候覺得很是麻煩,為什么定義時需要而調(diào)用時又不需要,為什么不能內(nèi)部簡化從而減少我們敲擊鍵盤的次數(shù)?你看完這篇文章后就會明白所有的疑問。
self代表類的實例,而非類。
實例來說明:
class Test:
def prt(self):
print(self)
print(self.__class__)
t = Test()
t.prt()
執(zhí)行結(jié)果如下
<__main__.Test object at 0x000000000284E080> <class '__main__.Test'>
從上面的例子中可以很明顯的看出,self代表的是類的實例。而self.class則指向類。
self不必非寫成self
有很多童鞋是先學(xué)習(xí)別的語言然后學(xué)習(xí)Python的,所以總覺得self怪怪的,想寫成this,可以嗎?
當(dāng)然可以,還是把上面的代碼改寫一下。
class Test:
def prt(this):
print(this)
print(this.__class__)
t = Test()
t.prt()
改成this后,運行結(jié)果完全一樣。
當(dāng)然,最好還是尊重約定俗成的習(xí)慣,使用self。
self可以不寫嗎
在Python的解釋器內(nèi)部,當(dāng)我們調(diào)用t.prt()時,實際上Python解釋成Test.prt(t),也就是說把self替換成類的實例。
有興趣的童鞋可以把上面的t.prt()一行改寫一下,運行后的實際結(jié)果完全相同。
實際上已經(jīng)部分說明了self在定義時不可以省略,如果非要試一下,那么請看下面:
class Test:
def prt():
print(self)
t = Test()
t.prt()
運行時提醒錯誤如下:prt在定義時沒有參數(shù),但是我們運行時強(qiáng)行傳了一個參數(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
當(dāng)然,如果我們的定義和調(diào)用時均不傳類實例是可以的,這就是類方法。
class Test:
def prt():
print(__class__)
Test.prt()
運行結(jié)果如下
<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()
運行結(jié)果如下
<__main__.Child object at 0x0000000002A47080> <__main__.Child object at 0x0000000002A47080> <__main__.Parent object at 0x0000000002A47240>
解釋:
運行c.cprt()時應(yīng)該沒有理解問題,指的是Child類的實例。
但是在運行c.pprt()時,等同于Child.pprt(c),所以self指的依然是Child類的實例,由于self中沒有定義pprt()方法,所以沿著繼承樹往上找,發(fā)現(xiàn)在父類Parent中定義了pprt()方法,所以就會成功調(diào)用。
在描述符類中,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
運行結(jié)果如下:
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不是應(yīng)該是調(diào)用它的實例t嗎?怎么變成了Desc類的實例了呢?
注意:此處需要睜大眼睛看清楚了,這里調(diào)用的是t.x,也就是說是Test類的實例t的屬性x,由于實例t中并沒有定義屬性x,所以找到了類屬性x,而該屬性是描述符屬性,為Desc類的實例而已,所以此處并沒有頂用Test的任何方法。
那么我們?nèi)绻苯油ㄟ^類來調(diào)用屬性x也可以得到相同的結(jié)果。
下面是把t.x改為Test.x運行的結(jié)果。
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'>
題外話:由于在很多時候描述符類中仍然需要知道調(diào)用該描述符的實例是誰,所以在描述符類中存在第二個參數(shù)ins,用來表示調(diào)用它的類實例,所以t.x時可以看到第三行中的運行結(jié)果中第二項為<main.Test object at 0x0000000002A570B8>。而采用Test.x進(jìn)行調(diào)用時,由于沒有實例,所以返回None。
從OO的本質(zhì)理解python中的self
舉個栗子,假設(shè)我要對用戶的數(shù)據(jù)進(jìn)行操作,用戶的數(shù)據(jù)包含name和age。如果用面向過程的話,實現(xiàn)出來是下面這樣子的。
def user_init(user,name,age):
user['name'] = name
user['age'] = age
def set_user_name(user, x):
user['name'] = x
def set_user_age(user, x):
user['age'] = x
def get_user_name(user):
return user['name']
def get_user_age(user):
return user['age']
myself = {}
user_init(myself,'kzc',17)
print get_user_age(myself)
set_user_age(myself,20)
print get_user_age(myself)
可以看到,對用戶的各種操作,都要傳user參數(shù)進(jìn)去。
如果用面向?qū)ο蟮脑?,就不用每次把user參數(shù)傳來傳去,把相關(guān)的數(shù)據(jù)和操作綁定在一個地方,在這個類的各個地方,可以方便的獲取數(shù)據(jù)。
之所以可以在類中的各個地方訪問數(shù)據(jù),本質(zhì)就是綁定了self這個東西,它方法的第一個參數(shù),當(dāng)然可以不叫self,叫其它名字,self只不過是個約定。
下面是面向?qū)ο蟮膶崿F(xiàn),可以看到,結(jié)構(gòu)化多了,清晰可讀。
class User(object):
def __init__(self,name,age):
self.name = name
self.age = age
def SetName(self,name):
self.name = name
def SetAge(self,age):
self.age = age
def GetName(self):
return self.name
def GetAge(self):
return self.age
u = User('kzc',17)
print u.GetName()
print u.GetAge()
從上面這個例子可以看出,其實面向?qū)ο笸τ杏玫?,只不過大多數(shù)人抽象的不好,封裝的不好,錯誤的運用。
總結(jié)
- self在定義時需要定義,但是在調(diào)用時會自動傳入。
- self的名字并不是規(guī)定死的,但是最好還是按照約定是用self
- self總是指調(diào)用時的類的實例。
相關(guān)文章
使用Python實現(xiàn)炫酷的數(shù)據(jù)動態(tài)圖大全
數(shù)據(jù)可視化是通過圖形、圖表、地圖等可視元素將數(shù)據(jù)呈現(xiàn)出來,以便更容易理解、分析和解釋,它是將抽象的數(shù)據(jù)轉(zhuǎn)化為直觀形象的過程,本文給大家介紹了使用Python實現(xiàn)炫酷的數(shù)據(jù)動態(tài)圖大全,需要的朋友可以參考下2024-06-06
Python 3.x 判斷 dict 是否包含某鍵值的實例講解
今天小編就為大家分享一篇Python 3.x 判斷 dict 是否包含某鍵值的實例講解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-07-07
python實現(xiàn)Web請求與響應(yīng)超詳細(xì)指南
Web請求是客戶端向服務(wù)器發(fā)起的資源獲取或操作請求,這篇文章主要介紹了python實現(xiàn)Web請求與響應(yīng)的相關(guān)資料,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下2025-05-05
Python實現(xiàn)PDF轉(zhuǎn)Word的多種方式總結(jié)
這篇文章主要為大家詳細(xì)介紹了三種Python實現(xiàn)PDF文件轉(zhuǎn)Word文檔的方式,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-11-11

