Python魔術(shù)方法專題
_del_
類的析構(gòu)方法,它在對象被回收時(shí)執(zhí)行,主要的作用時(shí)用來釋放資源(內(nèi)存 文件 進(jìn)程等)
因?yàn)镻ython內(nèi)存回收機(jī)制,使得Python的del方法的執(zhí)行時(shí)間是不確定的,因此不推薦在Python中使用析構(gòu)方法。
class Bar(object): def __del__(self): print("被回收了! ~") a = Bar() a.__del__() # 主動(dòng)調(diào)用是沒用的,因?yàn)橐糜?jì)數(shù)不為零,并不會(huì)回收資源 gc print("已經(jīng)刪除a了") print(a) del a # print(a)
_dict_
- 是一個(gè)綁定對象屬性的字典 存儲(chǔ)的是屬性的 鍵值對應(yīng)關(guān)系
- 可以直接通過修改這個(gè)字典來為對象添加屬性(但是不推薦這樣做!會(huì)使得程序的可讀性降低 破壞程序的結(jié)構(gòu) 充分理解 后使用 但是也要慎重)甚至 你可以通過修改 dict 來為對象添加方法 例如 func
_slots_
- 限定類的對象只能擁有某些屬性,防止寫錯(cuò)屬性名,也可以實(shí)現(xiàn)不允許動(dòng)態(tài)添加其他屬性。
- 形式:一個(gè)元組或 列表
- 需要注意 一旦類指定了 slots 那就意味著 類的屬性鍵值綁定關(guān)系 由__slots__來維護(hù) 也就是說 對象將沒有 __dict__方法
- __slots__只能約束本類,不能約束繼承它的子類,如果子類也定義了slots 方法,那么對子類的約束將會(huì)成為兩者的并集。
class Bar(object): __slots__ = ('name', 'gender') def __init__(self, name='monkey'): self.name = name self.gender = 'male' a = Bar() a.age = 18 # 動(dòng)態(tài)添加屬性是會(huì)報(bào)錯(cuò)的。 print(a.name)
_str_
必須返回一個(gè)str 類型 在打印對象的時(shí)候?qū)?huì) 打印返回的 str 而不是默認(rèn)的 self.str
:return: <main.... object at 0x1084b7208>
class Bar(object): def __str__(self): return "Bar" a = Bar() print(a) # Bar
_repr_
將對象轉(zhuǎn)化成對解釋器友好的形式,它跟eval()方法聯(lián)系緊密,通常repr()調(diào)用 對象的__repr__方法,該方法返回以字符串格式的 對解釋器友好的 對象描述,eval() 可以將repr()的返回值 轉(zhuǎn)化為原對象。
這玩意很強(qiáng)大,它是最直接的多態(tài)體現(xiàn),幾乎任何類對象都實(shí)現(xiàn)了它,但是每個(gè)返回的結(jié)果都是不一樣的。
_class_
_class_ 允許通過對象調(diào)用類的方法和操作類的屬性即 object.__class__ 可以拿到這個(gè)對象的類
拿到類后可以進(jìn)行新的實(shí)例化 操作類的屬性 調(diào)用類的方法等.
class Bar(object): name = 'monkey' a = Bar() print(a.__class__.name) # 允許通過實(shí)例化對象訪問類
_doc_
打印對象或類或方法的文檔字符串
class Bar(object): """ A simple show class! """ name = 'monkey' def get_name(self): """ get class argument name """ return self.__class__.name a = Bar() print(a.__class__.__doc__) print(a.__class__.get_name.__doc__) # A simple show class! # # # get class argument name
_base_
用來返回類的父類
_bases_
用來返回類的繼承列表
class Lady(object): """ """ class Small(object): """ """ class SmallLady(Small, Lady): """""" print(Lady.__base__) # <class 'object'> print(SmallLady.__bases__) # (<class '__main__.Small'>, <class '__main__.Lady'>)
_iter_
必須返回可迭代對象
這個(gè)對象需要實(shí)現(xiàn)__next__方法。
_next_
每次返回迭代器的下一個(gè)值或一個(gè)迭代異常來終止迭代。
_len_
每次返回迭代器的下一個(gè)值或一個(gè)迭代異常來終止迭代。
class ListMeta(type): def __call__(self, data, *args, **kwargs): # 使得self 也就是實(shí)例化出的類 是可調(diào)用的 List() 這里的self指的是 將要 實(shí)例化出來的類 本身 self.__init__(self,data) return self def __str__(self): result = self.clean_data(self) # 是 List 可以返回期望的列表格式 將對象轉(zhuǎn)化為對人友好的字符串 result = '[{}]'.format(result[:-1]) return result def __repr__(self): return 'List({})'.format(self.__str__()) # 轉(zhuǎn)化為對解釋器友好的字符串 def __iter__(self): # 返回實(shí)現(xiàn)了迭代器協(xié)議的對象 return self # 它本身實(shí)現(xiàn)了 __next__ def __next__(self): # 實(shí)現(xiàn)迭代器協(xié)議,每次返回下一個(gè)值 或 一個(gè)迭代異常終止迭代 if self.index >= len(self.data): raise StopIteration else: value = self.data[self.index] self.index += 1 return value def __len__(self): # 返回對象的長度,len()函數(shù)會(huì)執(zhí)行對象的 __len__方法 return self.len class List(metaclass=ListMeta): def __init__(self, data): self.data = data self.index = 0 self.len = len(self.data) l = List([1,2,3,4,5,6,7]) print(l) print(len(l)) for i in l: print(i)
_hash_
必須返回一個(gè)int類型的數(shù)據(jù),并且可以唯一的表示這個(gè)對象。這點(diǎn)很重要。
_getattribute_
- 此方法在每次訪問對象的屬性之前都會(huì)被調(diào)用,它容易使你陷入無限的遞歸中。
- 如果需要對對象屬性的訪問做一些限制 譬如 以"block_" 開頭的屬性不允許訪問可以這樣來實(shí)現(xiàn),這時(shí)候她是非常有用的。
- 如果該方法找到了對象的屬性,那么直接返回其屬性值,如果找不到或報(bào)錯(cuò)了,無論如何沒有達(dá)到預(yù)期的結(jié)果,那就調(diào)用 _getattr_ 方法。
_getattr_
- 當(dāng)以 點(diǎn) 屬性名的形式訪問屬性時(shí),如果屬性不存在,則會(huì)執(zhí)行對象的 _getattr_ 方法 該方法接受一個(gè)變量,item,即訪問的屬性名。返回值為本次獲取的屬性值,但是這個(gè)值并沒有寫入 對象的屬性字典里。
- 也就是說如果屬性在__getattribute__中找到是不會(huì)執(zhí)行這個(gè)方法的。
- 這個(gè)方法也容易陷入無限的遞歸當(dāng)中。
_setattr_
以點(diǎn)屬性名的形式設(shè)置屬性時(shí),會(huì)調(diào)用 _setattr_ 方法,此方法需要將屬性名和屬性值的對應(yīng)關(guān)系寫入關(guān)系字典__dict__里。如果重寫了該方法,一定不要忘記手動(dòng)的更新 對象屬性字典。
class Storage(object): def __init__(self, name): self.name = name # 調(diào)用__setattr__方法 def __getattribute__(self, item): # 每個(gè)屬性訪問前都先調(diào)用該方法 print('getattribute: %s' % item) ret = True if item == 'error': raise AttributeError(r'Error ~ "error"') # 報(bào)錯(cuò)了依然執(zhí)行~ else: ret = object.__getattribute__(self, item) return ret def __getattr__(self, item): print('getattr: %s' % item) try: return self.__dict__[item] except (IndexError, KeyError)as e: print('No attribute %s ' % e) return '%s is error' % item def __setattr__(self, key, value): print('setattr: %s ' % key) self.__dict__.update({key:value}) file = Storage('file') name = file.error # 調(diào)用 __getattr__ 方法 # setattr: name # getattribute: __dict__ # getattribute: error # getattr: error # getattribute: __dict__ # No attribute 'error'
以上就是Python魔術(shù)方法專題的詳細(xì)內(nèi)容,更多關(guān)于Python 魔術(shù)方法的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
解決安裝python庫時(shí)windows error5 報(bào)錯(cuò)的問題
今天小編就為大家分享一篇解決安裝python庫時(shí)windows error5 報(bào)錯(cuò)的問題,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-10-10python使用PIL和matplotlib獲取圖片像素點(diǎn)并合并解析
這篇文章主要介紹了python使用PIL和matplotlib獲取圖片像素點(diǎn)并合并解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-09-09Python批量自動(dòng)修改文件名,按指定的格式自動(dòng)命名方式
這篇文章主要介紹了Python批量自動(dòng)修改文件名,按指定的格式自動(dòng)命名方式,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-08-08Python實(shí)現(xiàn)從Markdown到PDF的轉(zhuǎn)換的方法
Markdown,以其簡潔的語法和易于閱讀的特性,成為了許多作家、開發(fā)者和學(xué)生記錄思想、編寫教程或撰寫報(bào)告的首選格式,然而,在分享或打印這些文檔時(shí),Markdown的純文本形式可能無法滿足對版式和布局的專業(yè)需求,本文將介紹如何用Python代碼輕松實(shí)現(xiàn)從Markdown到PDF的轉(zhuǎn)換2024-07-07