Python魔法方法 容器部方法詳解
這篇文章主要介紹了Python魔法方法 容器部方法詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
為了加深印象,也為了以后能夠更好的回憶,還是記錄一下。
序列(類似集合,列表,字符串),映射(類似字典)基本上是元素的集合,要實(shí)現(xiàn)他們的基本行為(協(xié)議),不可變對象需要兩個(gè)協(xié)議,可變對象需要4個(gè)協(xié)議。
- __len__(self):返回元素的數(shù)量,(為不可變對象需要的協(xié)議之一)=====> len
- __iter__返回一個(gè)迭代器,具有了__next__方法后,給for使用。
- __contains__ 代表 in的意思 xx.__contains__ (22) ==>22 in xx一個(gè)效果
- __getitem__(self, key)或者_(dá)_getitem__(self, index), 返回執(zhí)行輸入所關(guān)聯(lián)的值(為不可變對象需要的協(xié)議之一)
- __setitem__(self, key, values) 或者 __setitem__(self, index, values) , 設(shè)置指定輸入的值對應(yīng)的values
- __delitem__ (self, key) 刪除指定key的值
- __missing__這個(gè)有意思,跟__getattr__有的一比,是找不到這個(gè)key,觸發(fā)條件。前面用列表測試了,暈死了(只對字典有效。)
- __del__, 析構(gòu)函數(shù)當(dāng)這個(gè)類不存在實(shí)例對象時(shí)執(zhí)行。
下面我編寫一個(gè)自定義類似列表的類,實(shí)例后該類默認(rèn)前面有10個(gè)None參數(shù),且不能刪除前面5個(gè)空None。(隨口說的,開始寫了)
def check_index(index): if index < 5: raise IndexError('index must greater than 10') class S_List: def __init__(self): self.ll = [None] * 10 def __len__(self): # 提取參數(shù)長度 return len(self.ll) def __getitem__(self, index): # 取出參數(shù) return self.ll[index] def __setitem__(self, index, value): # 設(shè)置參數(shù) check_index(index) self.ll[index] = value def __delitem__(self, index): check_index(index) self.ll.pop(index) def __str__(self): # 打印對象時(shí),輸出列表本身 return str(self.ll) def __del__(self): # 沒有手工刪除在程序結(jié)束時(shí)釋放 print('我被釋放了!') sl = S_List() del sl[3] print(isinstance(sl, S_List)) print(f'輸出原始數(shù)據(jù):{sl}') sl[6] = 'six' print(f'修改后原始數(shù)據(jù):{sl}') print(f'隨便取一個(gè)值:{sl[1]}') del sl[6] print(f'第二次修改后原始數(shù)據(jù):{sl}') del sl[3] # sl[4] = 'oh' print(sl)
正常輸出:
True 輸出原始數(shù)據(jù):[None, None, None, None, None, None, None, None, None, None] 修改后原始數(shù)據(jù):[None, None, None, None, None, None, 'six', None, None, None] 隨便取一個(gè)值:None 第二次修改后原始數(shù)據(jù):[None, None, None, None, None, None, None, None, None] [None, None, None, None, None, None, None, None, None] 我被釋放了!
報(bào)錯(cuò)提示:
Traceback (most recent call last): File "/Users/shijianzhong/Desktop/yunzuan_buy/study_base.py", line 81, in <module> del sl[3] File "/Users/shijianzhong/Desktop/yunzuan_buy/study_base.py", line 73, in __delitem__ check_index(index) File "/Users/shijianzhong/Desktop/yunzuan_buy/study_base.py", line 53, in check_index raise IndexError('index must greater than 10') IndexError: index must greater than 10 我被釋放了!
這個(gè)是自定義的一個(gè)基本沒有什么方法的偽字典,不能增加元素,而且index,count等方法由于沒有寫入都無法使用。
好的方式是可以繼承l(wèi)ist或者dict的類,在里面對需要的條件進(jìn)行修改限制,這樣的話,實(shí)例出來的對象可以繼承原來的全部方法。
插入一個(gè)直接不用初始化自定義變量,直接借用__dict__來實(shí)現(xiàn)偽字典型的取值復(fù)制。
class Ii: def __getitem__(self, item): return self.__dict__[item] def __setitem__(self, key, value): self.__dict__[key] = value li = Ii() li[3] = 5 print(li[3]) # 5
這次我可以正真的定義個(gè)超級列表,根據(jù)我的需要。現(xiàn)在要求這個(gè)列表初始化有5個(gè)None元素,前面5個(gè)元素不能修改,后面添加的元素必須為str
def check_str(params): if not isinstance(params, str): raise ValueError('parameters must is string') def check_index(index): if index < 5: raise IndexError('index must greater than 10') class Super_List(list): def __init__(self): super(Super_List, self).__init__() # 調(diào)用父類初始化 self += [None] * 5 # 對初始化的參數(shù)進(jìn)行修改 def append(self, *args): # 對append進(jìn)行參數(shù)限制 for i in args: check_str(i) return super(Super_List, self).append(*args) def insert(self, index, *args): # 對insert的參數(shù)(索引及插入元素)進(jìn)行限制 check_index(index) # 判斷插入位置 for i in args: check_str(i) return super(Super_List, self).insert(index, *args) def extend(self, *args): # 對擴(kuò)張的列表元素進(jìn)行判斷 temp = args[0] for i in temp: check_str(i) super(Super_List, self).extend(*args) def __delitem__(self, index): # 對del命令的索引進(jìn)行判斷 check_index(index) super(Super_List, self).__delitem__(index) def clear(self): # 禁止使用clear命令 raise TypeError('No permission') ss_l = Super_List() print(ss_l) ss_l.append('1') ss_l.insert(5, 'a') ss_l.extend(['a', 'b', 'c']) ss_l.clear() print(ss_l)
寫了快半個(gè)小時(shí),感覺列表的增加與刪除命令很多,所有有一些命令沒有重寫,但邏輯還是一樣的。
如果向在有人訪問參數(shù)的時(shí)候,自動(dòng)執(zhí)行某些命令,可以寫在__getitem__下面。
跟新后,添加一個(gè)__missing__感覺還是非常有意思的。
class Dict(dict): def __init__(self, *args, **kwargs): # self.x = 12 super(Dict, self).__init__(*args, **kwargs) def __missing__(self, key): self[key] = None return self[key] l = Dict(((1,2),(2,3))) print(l) print(l[8]) print(l)
{1: 2, 2: 3} None {1: 2, 2: 3, 8: None}
有點(diǎn)像字典的內(nèi)置方式setdefault,我看能不能改成一樣的。
已經(jīng)寫完了,通過[]取值。
# -*- coding: utf-8 -*- class Dict(dict): def __init__(self, *args, **kwargs): super(Dict, self).__init__(*args, **kwargs) def __missing__(self, item): # 判斷進(jìn)來的參數(shù)是不是字符串,如果是字符串說明就是對象主動(dòng)調(diào)用__missing__進(jìn)來的 # 非__getitem__導(dǎo)入的 if isinstance(item, str): self[item] = 'Default Empty' return self[item] # 如果對象非字符串,明顯說明是__getitem__導(dǎo)入的,判斷長度就可以 else: key, value = item self[key] = value # 自身進(jìn)行賦值 return self[key] # 返回value def __getitem__(self, item): if not isinstance(item, tuple): # 傳進(jìn)來的item進(jìn)行判斷,如果非元祖,直接調(diào)用父類綁定self方法返回 return super(Dict, self).__getitem__(item) elif len(item) == 2 and isinstance(item, tuple): # 如果是元祖,又是2位長度的,進(jìn)行賦值。其實(shí)感覺元祖判斷沒有好像也沒關(guān)系 k, _ = item if k in self: return super(Dict, self).__getitem__(k) # 如果k在self里面繼續(xù)調(diào)用父類綁定self方法返回 else: res = self.__missing__(item) # 否則調(diào)用自身的__missing return res else: raise TypeError('input pattern error') # 元素?cái)?shù)量超過2個(gè),直接報(bào)錯(cuò) l = Dict((('name','sidian'),('age',99))) print(l) print(l['name','wudian']) print(l['addr','杭州']) print(l['hobby']) print(l)
{'name': 'sidian', 'age': 99} sidian 杭州 Default Empty {'name': 'sidian', 'age': 99, 'addr': '杭州', 'hobby': 'Default Empty'}
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Python實(shí)現(xiàn)元素等待代碼實(shí)例
這篇文章主要介紹了python實(shí)現(xiàn)元素等待代碼實(shí)例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-11-11python開發(fā)實(shí)例之Python的Twisted框架中Deferred對象的詳細(xì)用法與實(shí)例
這篇文章主要介紹了python開發(fā)實(shí)例之Python的Twisted框架中Deferred對象的詳細(xì)用法與實(shí)例,需要的朋友可以參考下2020-03-03對Python中列表和數(shù)組的賦值,淺拷貝和深拷貝的實(shí)例講解
今天小編就為大家分享一篇對Python中列表和數(shù)組的賦值,淺拷貝和深拷貝的實(shí)例講解,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-06-06pyinstaller打包單個(gè)exe后無法執(zhí)行錯(cuò)誤的解決方法
今天小編就為大家分享一篇pyinstaller打包單個(gè)exe后無法執(zhí)行錯(cuò)誤的解決方法,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-06-06Pycharm使用時(shí)會(huì)出現(xiàn)的問題之cv2無法安裝解決
這篇文章主要介紹了Pycharm使用時(shí)會(huì)出現(xiàn)的問題之cv2無法安裝解決方案,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-05-05深入解析Python中BeautifulSoup4的基礎(chǔ)知識(shí)與實(shí)戰(zhàn)應(yīng)用
BeautifulSoup4正是一款功能強(qiáng)大的解析器,能夠輕松解析HTML和XML文檔,本文將介紹BeautifulSoup4的基礎(chǔ)知識(shí),并通過實(shí)際代碼示例進(jìn)行演示,感興趣的可以了解下2024-02-02Python實(shí)現(xiàn)感知機(jī)(PLA)算法
這篇文章主要為大家詳細(xì)介紹了Python實(shí)現(xiàn)感知機(jī)(PLA)算法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-12-12