python學(xué)習(xí)-type()效斑,isinstance(),dir(),getattr(),setattr(),hasattr()


isinstance() 與 type() 區(qū)別:
type() 不會(huì)認(rèn)為子類是一種父類類型,不考慮繼承關(guān)系柱徙。
isinstance() 會(huì)認(rèn)為子類是一種父類類型缓屠,考慮繼承關(guān)系奇昙。
如果要判斷兩個(gè)類型是否相同推薦使用 isinstance()。

使用type()

首先敌完,我們來判斷對象類型储耐,使用type()函數(shù):

基本類型都可以用type()判斷:

>>> type(123)
<class 'int'>
>>> type('str')
<class 'str'>
>>> type(None)
<type(None) 'NoneType'>

如果一個(gè)變量指向函數(shù)或者類,也可以用type()判斷:

>>> type(abs)
<class 'builtin_function_or_method'>
>>> type(a)
<class '__main__.Animal'>

但是type()函數(shù)返回的是什么類型呢滨溉?它返回對應(yīng)的Class類型什湘。如果我們要在if語句中判斷,就需要比較兩個(gè)變量的type類型是否相同:

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

判斷基本數(shù)據(jù)類型可以直接寫int晦攒,str等闽撤,但如果要判斷一個(gè)對象是否是函數(shù)怎么辦晓猛?可以使用types模塊中定義的常量:

>>> 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()

對于class的繼承關(guān)系來說锅棕,使用type()就很不方便毒姨。我們要判斷class的類型枝缔,可以使用isinstance()函數(shù)右核。

我們回顧上次的例子同窘,如果繼承關(guān)系是:

object -> Animal -> Dog -> Husky

那么殖氏,isinstance()就可以告訴我們众雷,一個(gè)對象是否是某種類型讼庇。先創(chuàng)建3種類型的對象:

>>> a = Animal()
>>> d = Dog()
>>> h = Husky()

然后绎巨,判斷:

>>> isinstance(h, Husky)
True

沒有問題,因?yàn)?code>h變量指向的就是Husky對象蠕啄。

再判斷:

>>> isinstance(h, Dog)
True

h雖然自身是Husky類型场勤,但由于Husky是從Dog繼承下來的,所以歼跟,h也還是Dog類型和媳。換句話說,isinstance()判斷的是一個(gè)對象是否是該類型本身哈街,或者位于該類型的父繼承鏈上留瞳。

因此,我們可以確信骚秦,h還是Animal類型:

>>> isinstance(h, Animal)
True

同理她倘,實(shí)際類型是Dog的d也是Animal類型:

>>> isinstance(d, Dog) and isinstance(d, Animal)
True

但是,d不是Husky類型:

>>> isinstance(d, Husky)
False

能用type()判斷的基本類型也可以用isinstance()判斷:

>>> isinstance('a', str)
True
>>> isinstance(123, int)
True
>>> isinstance(b'a', bytes)
True

并且還可以判斷一個(gè)變量是否是某些類型中的一種作箍,比如下面的代碼就可以判斷是否是list或者tuple:

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

總是優(yōu)先使用isinstance()判斷類型硬梁,可以將指定類型及其子類“一網(wǎng)打盡”。

使用dir()

如果要獲得一個(gè)對象的所有屬性和方法胞得,可以使用dir()函數(shù)荧止,它返回一個(gè)包含字符串的list,比如,獲得一個(gè)str對象的所有屬性和方法:

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

類似__xxx__的屬性和方法在Python中都是有特殊用途的罩息,比如__len__方法返回長度嗤详。在Python中,如果你調(diào)用len()函數(shù)試圖獲取一個(gè)對象的長度瓷炮,實(shí)際上葱色,在len()函數(shù)內(nèi)部,它自動(dòng)去調(diào)用該對象的__len__()方法娘香,所以苍狰,下面的代碼是等價(jià)的:

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

我們自己寫的類,如果也想用len(myObj)的話烘绽,就自己寫一個(gè)__len__()方法:

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

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

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

僅僅把屬性和方法列出來是不夠的,配合getattr()安接、setattr()以及hasattr()翔忽,我們可以直接操作一個(gè)對象的狀態(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è)置一個(gè)屬性'y'
>>> hasattr(obj, 'y') # 有屬性'y'嗎?
True
>>> getattr(obj, 'y') # 獲取屬性'y'
19
>>> obj.y # 獲取屬性'y'
19

如果試圖獲取不存在的屬性胡野,會(huì)拋出AttributeError的錯(cuò)誤:

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

可以傳入一個(gè)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

小結(jié)

通過內(nèi)置的一系列函數(shù)熊响,我們可以對任意一個(gè)Python對象進(jìn)行剖析旨别,拿到其內(nèi)部的數(shù)據(jù)。要注意的是汗茄,只有在不知道對象信息的時(shí)候昼榛,我們才會(huì)去獲取對象信息。如果可以直接寫:

sum = obj.x + obj.y

就不要寫:

sum = getattr(obj, 'x') + getattr(obj, 'y')

一個(gè)正確的用法的例子如下:

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

假設(shè)我們希望從文件流fp中讀取圖像剔难,我們首先要判斷該fp對象是否存在read方法,如果存在奥喻,則該對象是一個(gè)流偶宫,如果不存在,則無法讀取环鲤。hasattr()就派上了用場纯趋。

請注意,在Python這類動(dòng)態(tài)語言中,根據(jù)鴨子類型吵冒,有read()方法纯命,不代表該fp對象就是一個(gè)文件流,它也可能是網(wǎng)絡(luò)流痹栖,也可能是內(nèi)存中的一個(gè)字節(jié)流亿汞,但只要read()方法返回的是有效的圖像數(shù)據(jù),就不影響讀取圖像的功能揪阿。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末疗我,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子南捂,更是在濱河造成了極大的恐慌吴裤,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,657評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件溺健,死亡現(xiàn)場離奇詭異麦牺,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)鞭缭,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,889評論 3 394
  • 文/潘曉璐 我一進(jìn)店門剖膳,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人缚去,你說我怎么就攤上這事潮秘。” “怎么了易结?”我有些...
    開封第一講書人閱讀 164,057評論 0 354
  • 文/不壞的土叔 我叫張陵枕荞,是天一觀的道長。 經(jīng)常有香客問我搞动,道長躏精,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,509評論 1 293
  • 正文 為了忘掉前任鹦肿,我火速辦了婚禮矗烛,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘箩溃。我一直安慰自己瞭吃,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,562評論 6 392
  • 文/花漫 我一把揭開白布涣旨。 她就那樣靜靜地躺著歪架,像睡著了一般。 火紅的嫁衣襯著肌膚如雪霹陡。 梳的紋絲不亂的頭發(fā)上和蚪,一...
    開封第一講書人閱讀 51,443評論 1 302
  • 那天止状,我揣著相機(jī)與錄音,去河邊找鬼攒霹。 笑死怯疤,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的催束。 我是一名探鬼主播集峦,決...
    沈念sama閱讀 40,251評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼泣崩!你這毒婦竟也來了少梁?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,129評論 0 276
  • 序言:老撾萬榮一對情侶失蹤矫付,失蹤者是張志新(化名)和其女友劉穎凯沪,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體买优,經(jīng)...
    沈念sama閱讀 45,561評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡妨马,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,779評論 3 335
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了杀赢。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片烘跺。...
    茶點(diǎn)故事閱讀 39,902評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖脂崔,靈堂內(nèi)的尸體忽然破棺而出滤淳,到底是詐尸還是另有隱情,我是刑警寧澤砌左,帶...
    沈念sama閱讀 35,621評論 5 345
  • 正文 年R本政府宣布脖咐,位于F島的核電站,受9級(jí)特大地震影響汇歹,放射性物質(zhì)發(fā)生泄漏屁擅。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,220評論 3 328
  • 文/蒙蒙 一产弹、第九天 我趴在偏房一處隱蔽的房頂上張望派歌。 院中可真熱鬧,春花似錦痰哨、人聲如沸胶果。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,838評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽稽物。三九已至,卻和暖如春折欠,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,971評論 1 269
  • 我被黑心中介騙來泰國打工锐秦, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留咪奖,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,025評論 2 370
  • 正文 我出身青樓酱床,卻偏偏與公主長得像羊赵,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子扇谣,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,843評論 2 354

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

  • 要點(diǎn): 函數(shù)式編程:注意不是“函數(shù)編程”昧捷,多了一個(gè)“式” 模塊:如何使用模塊 面向?qū)ο缶幊蹋好嫦驅(qū)ο蟮母拍睢傩浴?..
    victorsungo閱讀 1,504評論 0 6
  • # 第一優(yōu)先級(jí)規(guī)則聲明: # 除了夢境罐寨,每一個(gè)意識(shí)主進(jìn)程都必須與一個(gè)身體參與的機(jī)械進(jìn)程相匹配靡挥,否則結(jié)束意識(shí)主進(jìn)程。...
    李洞BarryLi閱讀 3,858評論 0 1
  • 〇鸯绿、前言 本文共108張圖跋破,流量黨請慎重! 歷時(shí)1個(gè)半月瓶蝴,我把自己學(xué)習(xí)Python基礎(chǔ)知識(shí)的框架詳細(xì)梳理了一遍毒返。 ...
    Raxxie閱讀 18,954評論 17 410
  • Python進(jìn)階框架 希望大家喜歡,點(diǎn)贊哦首先感謝廖雪峰老師對于該課程的講解 一舷手、函數(shù)式編程 1.1 函數(shù)式編程簡...
    Gaolex閱讀 5,498評論 6 53
  • 只是單純的想記錄一件事情。因?yàn)楣ぷ髟蛐睿以诤芷h(yuǎn)的地方任教弟劲。那里的孩子們,沒有特別嬌氣姥芥,用我們經(jīng)常說的一句...
    檸檬安然閱讀 187評論 0 0