Python學(xué)習(xí)筆記08-面向?qū)ο缶幊?/h1>

熟悉面向?qū)ο缶幊蹋∣bject Oriented Programming)的同學(xué)都知道類是OOP中必不可少的元素更胖,下面我們來了解下Python中是如何定義和運(yùn)用類的呢薪贫,如下:

class Student(object):

    def __init__(self,name,score):
        self.name = name
        self.score = score

    def print_score(self):
        print('%s:%s' % (self.name,self,score))


bart = Student('Bart Simpson' , 59)
lisa = Student('Lisa Simpson' , 87)
bart.print_score()
lisa.print_score()

封裝繼承多態(tài)是面向?qū)ο蟮娜筇匦浴?/p>

實(shí)例

類(Class)實(shí)例(Instance)是面向?qū)ο蟮闹匾拍睿?code>類是抽象的模板溪掀,比如Student類螃诅,而實(shí)列是根據(jù)類創(chuàng)建出來的一個個具體的對象诲锹,每個對象有相同的方法但各自的數(shù)據(jù)可能不同凿可。

class Student(object):
    pass

class后面緊接著的是類名惑折,類名通常是首字母大寫,緊接著是(object)枯跑,表示該類從哪個類繼承而來惨驶,如果沒有合適的繼承則使用object類。
在實(shí)例化后敛助,依然可以綁定新的屬性粗卜,這點(diǎn)與C++等靜態(tài)語言不同,如下:

>>>bart = Student()
>>>bart.age = 17

類起到的是模板作用纳击,所以定義模板(類)時休建,我們在模板中通過__init__()定義一些必須的屬性。
__init__方法的第一個參數(shù)是self评疗,表示實(shí)例本身,通過向self綁定屬性來實(shí)現(xiàn)向實(shí)例綁定屬性茵烈。
有了__init__方法百匆,在創(chuàng)建實(shí)例的時候,就不能傳入空的參數(shù)了呜投,必須傳入與__init__方法匹配的參數(shù)加匈,但self不需要傳,Python解釋器自己會把實(shí)例變量傳進(jìn)去仑荐。
和普通的函數(shù)相比雕拼,在中定義的函數(shù)只有一點(diǎn)不同,就是第一個參數(shù)永遠(yuǎn)是實(shí)例變量self粘招,并且啥寇,調(diào)用時,不用傳遞該參數(shù)。除此之外辑甜,類的方法和普通函數(shù)沒有什么區(qū)別衰絮,所以,你仍然可以用默認(rèn)參數(shù)磷醋、可變參數(shù)猫牡、關(guān)鍵字參數(shù)命名關(guān)鍵字參數(shù)。

訪問控制

在python中以兩個下劃線__開頭的屬性或方法為私用的(private)邓线。需要注意的是淌友,在Python中,變量名類似__xxx__的骇陈,也就是以雙下劃線開頭震庭,并且以雙下劃線結(jié)尾的,是特殊變量缩歪,特殊變量是可以直接訪問的归薛,不是private變量,所以匪蝙,不能用__name__主籍、__score__這樣的變量名。
單下劃線的屬性外部可以訪問逛球,但是約定俗成的規(guī)定是千元,可以把他當(dāng)作是私有變量,不要隨意訪問颤绕。
其實(shí)雙下劃線__變量也可以被訪問幸海,Python解釋器對外把__name變量改成了_Student__name。
最后注意下這種錯誤寫法:

>>> bart = Student('Bart Simpson', 59)
>>> bart.get_name()
'Bart Simpson'
>>> bart.__name = 'New Name' # 設(shè)置__name變量奥务!
>>> bart.__name
'New Name'

表面上看物独,外部代碼“成功”地設(shè)置了__name變量,但實(shí)際上這個__name變量和class內(nèi)部的__name變量不是一個變量氯葬!內(nèi)部的__name變量已經(jīng)被Python解釋器自動改成了_Student__name挡篓,而外部代碼給bart新增了一個__name變量。

繼承和多態(tài)

Python的繼承和多態(tài)與其他面向?qū)ο缶幊陶Z言沒大區(qū)別帚称,此處不再贅述官研,詳細(xì)了解可見廖雪峰官方網(wǎng)站
下面記錄下Python自有的語言特性:
當(dāng)我們定義一個class的時候闯睹,我們實(shí)際上就定義了一種數(shù)據(jù)類型戏羽。我們定義的數(shù)據(jù)類型和Python自帶的數(shù)據(jù)類型,比如str楼吃、list始花、dict沒什么兩樣:

a = list() # a是list類型
b = Animal() # b是Animal類型
c = Dog() # c是Dog類型

判斷一個變量是否是某個類型可以用isinstance()判斷:

>>> isinstance(a, list)
True
>>> isinstance(b, Animal)
True
>>> isinstance(c, Dog)
True
>>> isinstance(c, Animal)#此時也會返回True
True 

獲取對象信息

使用type()函數(shù)判斷對象類型:

>>> type(123)
<class 'int'>
>>> type('str')
<class 'str'>
>>> type(None)
<type(None) 'NoneType'>
#如果一個變量指向函數(shù)或者類妄讯,也可以用type()判斷:
>>> type(abs)
<class 'builtin_function_or_method'>
>>> type(a)
<class '__main__.Animal'>

type()返回的是對應(yīng)的Class類型,如果我們要在if語句中判斷衙荐,就需要比較兩個變量的type類型是否相同:

>>> type(123)==type(456)
True
>>> type(123)==int
True
>>> type('abc')==type('123')
True
>>> type('abc')==str
True
>>> type('abc')==type(123)
False

也可以判斷具體的類型:

>>> import types
>>> def fn():
...     pass
...
>>> type(fn)==types.FunctionType
True
>>> type(abs)==types.BuiltinFunctionType
True
>>> type(lambda x: x)==types.LambdaType
True
>>> type((x for x in range(10)))==types.GeneratorType
True

isinstance可以判斷一個變量是否是某些類型中的一種捞挥,比如下面的代碼就可以判斷是否是list或者tuple

>>> isinstance([1, 2, 3], (list, tuple))
True
>>> isinstance((1, 2, 3), (list, tuple))
True

dir()方法

dir()函數(shù)可以獲取一個對象的所有屬性和方法,它返回的是一個包含字符串的list忧吟,比如:

>>> dir('ABC')
['__add__', '__class__',..., '__subclasshook__', 'capitalize', 'casefold',..., 'zfill']

類似__xxx__的屬性和方法在Python中都是有特殊用途的砌函,比如__len__方法返回長度。在Python中溜族,如果你調(diào)用len()函數(shù)試圖獲取一個對象的長度讹俊,實(shí)際上,在len()函數(shù)內(nèi)部煌抒,它自動去調(diào)用該對象的__len__()方法仍劈,所以,下面的代碼是等價(jià)的:

>>> len('ABC')
3
>>> 'ABC'.__len__()
3

我們自己寫的類寡壮,如果也想用len(myObj)的話贩疙,就自己寫一個__len__()方法:

>>> class MyDog(object):
...     def __len__(self):
...         return 100
...
>>> dog = MyDog()
>>> len(dog)
100

剩下的都是普通屬性或方法,比如lower()返回小寫的字符串:

>>> 'ABC'.lower()
'abc'

僅僅把屬性和方法列出來是不夠的况既,配合getattr()这溅、setattr()以及hasattr(),我們可以直接操作一個對象的狀態(tài):

>>> class MyObject(object):
...     def __init__(self):
...         self.x = 9
...     def power(self):
...         return self.x * self.x
...
>>> obj = MyObject()
>>> hasattr(obj, 'x') # 有屬性'x'嗎棒仍?
True
>>> obj.x
9
>>> hasattr(obj, 'y') # 有屬性'y'嗎悲靴?
False
>>> setattr(obj, 'y', 19) # 設(shè)置一個屬性'y'
>>> hasattr(obj, 'y') # 有屬性'y'嗎?
True
>>> getattr(obj, 'y') # 獲取屬性'y'
19
>>> obj.y # 獲取屬性'y'
19

如果試圖獲取不存在的屬性莫其,會拋出AttributeError的錯誤:

>>> getattr(obj, 'z') # 獲取屬性'z'
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: 'MyObject' object has no attribute 'z'

可以傳入一個default參數(shù)癞尚,如果屬性不存在,就返回默認(rèn)值:

>>> getattr(obj, 'z', 404) # 獲取屬性'z'乱陡,如果不存在浇揩,返回默認(rèn)值404
404

也可以獲得對象的方法:

>>> hasattr(obj, 'power') # 有屬性'power'嗎?
True
>>> getattr(obj, 'power') # 獲取屬性'power'
<bound method MyObject.power of <__main__.MyObject object at 0x10077a6a0>>
>>> fn = getattr(obj, 'power') # 獲取屬性'power'并賦值到變量fn
>>> fn # fn指向obj.power
<bound method MyObject.power of <__main__.MyObject object at 0x10077a6a0>>
>>> fn() # 調(diào)用fn()與調(diào)用obj.power()是一樣的
81

一個hasattr()函數(shù)用法的例子:

def readImage(fp):
    if hasattr(fp, 'read'):
        return readData(fp)
    return None

假設(shè)我們希望從文件流fp中讀取圖像憨颠,我們首先要判斷該fp對象是否存在read方法临燃,如果存在,則該對象是一個流烙心,如果不存在,則無法讀取乏沸。hasattr()就派上了用場淫茵。
請注意,在Python這類動態(tài)語言中蹬跃,有read()方法匙瘪,不代表該fp對象就是一個文件流铆铆,它也可能是網(wǎng)絡(luò)流,也可能是內(nèi)存中的一個字節(jié)流丹喻,但只要read()方法返回的是有效的圖像數(shù)據(jù)薄货,就不影響讀取圖像的功能。

實(shí)例屬性和類屬性

由于Python是動態(tài)語言碍论,根據(jù)類創(chuàng)建的實(shí)例可以任意綁定屬性谅猾。
給實(shí)例綁定屬性的方法是通過實(shí)例變量,或者通過self變量:

class Student(object):
    def __init__(self, name):
        self.name = name

s = Student('Bob')
s.score = 90

也可以直接在class中定義屬性鳍悠,這種屬性是類屬性税娜,歸Student類所有:

class Student(object):
    name = 'Student'

詳見下面的例子:

>>> class Student(object):
...     name = 'Student'
...
>>> s = Student() # 創(chuàng)建實(shí)例s
>>> print(s.name) # 打印name屬性,因?yàn)閷?shí)例并沒有name屬性藏研,所以會繼續(xù)查找class的name屬性
Student
>>> print(Student.name) # 打印類的name屬性
Student
>>> s.name = 'Michael' # 給實(shí)例綁定name屬性
>>> print(s.name) # 由于實(shí)例屬性優(yōu)先級比類屬性高敬矩,因此,它會屏蔽掉類的name屬性
Michael
>>> print(Student.name) # 但是類屬性并未消失蠢挡,用Student.name仍然可以訪問
Student
>>> del s.name # 如果刪除實(shí)例的name屬性
>>> print(s.name) # 再次調(diào)用s.name弧岳,由于實(shí)例的name屬性沒有找到,類的name屬性就顯示出來了
Student

實(shí)例屬性屬于各個實(shí)例所有业踏,互不干擾禽炬;類屬性屬于所有,所有實(shí)例共享一個屬性堡称;

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者

  • 序言:七十年代末瞎抛,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子却紧,更是在濱河造成了極大的恐慌桐臊,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,743評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件晓殊,死亡現(xiàn)場離奇詭異断凶,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)巫俺,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,296評論 3 385
  • 文/潘曉璐 我一進(jìn)店門认烁,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人介汹,你說我怎么就攤上這事却嗡。” “怎么了嘹承?”我有些...
    開封第一講書人閱讀 157,285評論 0 348
  • 文/不壞的土叔 我叫張陵窗价,是天一觀的道長。 經(jīng)常有香客問我叹卷,道長撼港,這世上最難降的妖魔是什么坪它? 我笑而不...
    開封第一講書人閱讀 56,485評論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮帝牡,結(jié)果婚禮上往毡,老公的妹妹穿的比我還像新娘。我一直安慰自己靶溜,他們只是感情好开瞭,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,581評論 6 386
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著墨技,像睡著了一般惩阶。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上扣汪,一...
    開封第一講書人閱讀 49,821評論 1 290
  • 那天断楷,我揣著相機(jī)與錄音,去河邊找鬼崭别。 笑死冬筒,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的茅主。 我是一名探鬼主播舞痰,決...
    沈念sama閱讀 38,960評論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼诀姚!你這毒婦竟也來了响牛?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,719評論 0 266
  • 序言:老撾萬榮一對情侶失蹤赫段,失蹤者是張志新(化名)和其女友劉穎呀打,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體糯笙,經(jīng)...
    沈念sama閱讀 44,186評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡贬丛,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,516評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了给涕。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片豺憔。...
    茶點(diǎn)故事閱讀 38,650評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖够庙,靈堂內(nèi)的尸體忽然破棺而出恭应,到底是詐尸還是另有隱情,我是刑警寧澤耘眨,帶...
    沈念sama閱讀 34,329評論 4 330
  • 正文 年R本政府宣布暮屡,位于F島的核電站,受9級特大地震影響毅桃,放射性物質(zhì)發(fā)生泄漏褒纲。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,936評論 3 313
  • 文/蒙蒙 一钥飞、第九天 我趴在偏房一處隱蔽的房頂上張望莺掠。 院中可真熱鬧,春花似錦读宙、人聲如沸彻秆。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,757評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽唇兑。三九已至,卻和暖如春桦锄,著一層夾襖步出監(jiān)牢的瞬間扎附,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,991評論 1 266
  • 我被黑心中介騙來泰國打工结耀, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留留夜,地道東北人。 一個月前我還...
    沈念sama閱讀 46,370評論 2 360
  • 正文 我出身青樓图甜,卻偏偏與公主長得像碍粥,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子黑毅,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,527評論 2 349

推薦閱讀更多精彩內(nèi)容