Python?列表和字典常踩坑即解決方案
前言:
在Python中,如果我們想要在遍歷一組數(shù)據(jù)的過(guò)程中,對(duì)這組數(shù)據(jù)進(jìn)行修改,通常會(huì)出現(xiàn)許多問(wèn)題,例如對(duì)列表進(jìn)行上述操作時(shí), 會(huì)忽略部分?jǐn)?shù)據(jù);遍歷字典時(shí),不能修改數(shù)據(jù)。本文針對(duì)這些問(wèn)題,提出了多種解決方案。
一、關(guān)于列表
1.問(wèn)題描述
在Python中,如果你試圖在遍歷一組數(shù)據(jù)的過(guò)程中,對(duì)其進(jìn)行修改,這通常沒(méi)什么問(wèn)題。
例如:
l?=?[3,?4,?56,?7,?10,?9,?6,?5] for?i?in?l: ????if?not?i?%?2?==?0: ????????continue ????l.remove(i) print(l)
上述這段代碼遍歷了一個(gè)包含數(shù)字的列表,為了去除掉所有偶數(shù),直接修改了列表l。
然而,運(yùn)行后輸出卻是:
[3, 56, 7, 9, 5]
等一下!輸出似乎不對(duì)。最終的結(jié)果仍然含有一個(gè)偶數(shù)56。為什么沒(méi)有成功去除這個(gè)數(shù)呢?我們可以嘗試打印出 for循環(huán)遍歷的所有元素,
運(yùn)行如下代碼:
l?=?[3,?4,?56,?7,?10,?9,?6,?5] for?i?in?l: ????print(i) ????if?not?i?%?2?==?0: ????????continue ????l.remove(i) print(l)
這段代碼的輸出為:
3
4
7
10
6
[3, 56, 7, 9, 5]
從輸出可以看出,for循環(huán)似乎沒(méi)有訪問(wèn)列表中的所有元素。為了解for循環(huán)在內(nèi)部究竟做了什么, 我們可以使用 iter 和 next 來(lái)模擬一下。
看看下面這個(gè)例子,我使用了ipython shell 來(lái)運(yùn)行代碼:
In?[1]:?l?=?[3,?4,?56,?7,?10,?9,?6,?5] In?[2]:?#?把列表變成一個(gè)迭代器 In?[3]:?it?=?iter(l) In?[4]:?#?使用?next()?方法來(lái)模擬?for循環(huán) In?[5]:?next(it) Out[5]:?3 In?[6]:?next(it) Out[6]:?4 In?[7]:?#?移除一個(gè)迭代器已經(jīng)訪問(wèn)過(guò)的元素 In?[8]:?l.remove(3) In?[9]:?next(it) Out[9]:?7 In?[10]:?#?注意此處跳過(guò)了56,我們可以再移除一個(gè)元素 In?[11]:?l.remove(4) In?[12]:?next(it) Out[12]:?9
上面這個(gè)實(shí)驗(yàn)揭示了:當(dāng)你移除一個(gè)迭代器已經(jīng)訪問(wèn)過(guò)的元素后,在下一次迭代時(shí),會(huì)跳過(guò)右邊的一個(gè)元素,直接訪問(wèn)下一個(gè)。
反之依然成立,即當(dāng)開始迭代后,如果你在列表開頭添加了一個(gè)元素,下次迭代時(shí),可能會(huì)訪問(wèn)到已經(jīng)迭代過(guò)的元素,
下面這段代碼就出現(xiàn)了這種情況:
In[1]:?l?=?[3,?4,?56,?7,?10,?9,?6,?5] In[2]:?it?=?iter(l) In[3]:?next(it) Out[3]:?3 In[4]:?next(it) Out[4]:?4 In[5]:?l.insert(0,?44) In[6]:?next(it) Out[6]:?4
注意:當(dāng)在列表頭部添加了44后,4被訪問(wèn)了兩次。
2.解決方案
為了解決上述問(wèn)題,我們必須得確保:不能移除迭代器訪問(wèn)過(guò)的元素。
方案一
我們可以先對(duì)原列表進(jìn)行翻轉(zhuǎn)得到一個(gè)新列表,再對(duì)新列表進(jìn)行迭代,并在原列表 l 中移除不符合條件的元素。
該方案代碼如下:
l?=?[3,?4,?56,?7,?10,?9,?6,?5] #?迭代翻轉(zhuǎn)后的列表 for?i?in?reversed(l): ????print(i) ????if?not?i?%?2?==?0: ????????continue ????l.remove(i) print(l)
結(jié)果如下:
5
6
9
10
7
56
4
3
[3, 7, 9, 5]
注意:迭代器現(xiàn)在成功訪問(wèn)到了列表中的所有元素,并最終輸出了只含有奇數(shù)的列表。
方案二
我們還可以在開始迭代前,先復(fù)制列表 l 。但是當(dāng)列表 l 中的數(shù)據(jù)過(guò)多時(shí),這樣做顯然比較耗費(fèi)性能。
該方案代碼如下:
l?=?[3,?4,?56,?7,?10,?9,?6,?5] #?在這里使用?'l.copy()'?來(lái)對(duì)列表?l?進(jìn)行淺拷貝 for?i?in?l.copy():?? ????print(i)??? ????if?not?i?%?2?==?0:????? ????????continue?? ????l.remove(i) print(l)
輸出如下:
3
4
56
7
10
9
6
5
[3, 7, 9, 5]
該方案能保證迭代的順序和移除元素的順序相同。不過(guò)由于迭代和移除這兩種操作針對(duì)的是兩個(gè)不同的列表,因此順序相同并不重要。
二、關(guān)于字典
1.問(wèn)題描述
在對(duì)字典進(jìn)行迭代時(shí),不能修改字典。如下:
#?{0:?0,?1:?1,?2:?2,?3:?3,?4:?4,?5:?5,?6:?6,?7:?7,?8:?8,?9:?9} d?=?{k:?k?for?k?in?range(10)} for?k,?v?in?d.items():?? ????if?not?v?%?2?==?0:???? ????????continue?? ????d.pop(k)
這段代碼會(huì)產(chǎn)生 RuntimeError :
Traceback?(most?recent?call?last):?? ??File?"F:/Documents/pythonprojects/01practice/app.py",?line?7,?in?<module>?? ????for?k,?v?in?d.items(): RuntimeError:?dictionary?changed?size?during?iteration
2.解決方案
我們可以先復(fù)制字典的所有 key ,隨后在迭代 key 的過(guò)程中,移除不符合條件的元素。過(guò)程如下:
#?{0:?0,?1:?1,?2:?2,?3:?3,?4:?4,?5:?5,?6:?6,?7:?7,?8:?8,?9:?9} d?=?{k:?k?for?k?in?range(10)} #?這里復(fù)制了字典中的所有key值 #?沒(méi)有復(fù)制整個(gè)字典 #?同時(shí)使用tuple()速度更快 for?k?in?tuple(d.keys()):??? ????if?not?d[k]?%?2?==?0:???? ????????continue?? ????d.pop(k) ???? print(d)
運(yùn)行代碼后輸出如下:
{1: 1, 3: 3, 5: 5, 7: 7, 9: 9}
我們成功移除了字典中的所有偶數(shù)鍵值對(duì)!
結(jié)論
文中我們針對(duì)迭代一組數(shù)據(jù)時(shí)無(wú)法進(jìn)行修改的問(wèn)題,分別提出了不同的解決方案:如果想在遍歷列表的時(shí)候,對(duì)列表進(jìn)行修改, 我們可以先對(duì)原列表進(jìn)行翻轉(zhuǎn)或復(fù)制,從而得到一個(gè)新列表,隨后在遍歷新列表的過(guò)程中,修改原列表中的數(shù)據(jù);如果我們想在遍歷字典的時(shí)候,對(duì)字典進(jìn)行修改,可以先復(fù)制字典的所有鍵值,然后在迭代鍵值的時(shí)候,修改字典中的數(shù)據(jù)。
到此這篇關(guān)于 Python 列表和字典常踩坑即解決方案的文章就介紹到這了,更多相關(guān) Python 列表和字典 內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
python區(qū)塊鏈實(shí)現(xiàn)簡(jiǎn)版工作量證明
這篇文章主要為大家介紹了python區(qū)塊鏈實(shí)現(xiàn)簡(jiǎn)版工作量證明詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-05-05Python實(shí)現(xiàn)動(dòng)態(tài)加載模塊、類、函數(shù)的方法分析
這篇文章主要介紹了Python實(shí)現(xiàn)動(dòng)態(tài)加載模塊、類、函數(shù)的方法,結(jié)合實(shí)例形式分析了Python動(dòng)態(tài)加載模塊、類及函數(shù)的實(shí)現(xiàn)方法及操作技巧,需要的朋友可以參考下2017-07-07Python使用pandas處理CSV文件的實(shí)例講解
今天小編就為大家分享一篇Python使用pandas處理CSV文件的實(shí)例講解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-06-06使用Pyhton集合set()實(shí)現(xiàn)成果查漏的例子
今天小編就為大家分享一篇使用Pyhton集合set()實(shí)現(xiàn)成果查漏的例子,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-11-11一個(gè)小示例告訴你Python語(yǔ)言的優(yōu)雅之處
本篇中, 我們展示一下一段非常小的代碼, 這段代碼十分吸引我們, 因?yàn)樗褂檬謨?yōu)雅和直接的方式解決了一個(gè)常見(jiàn)的問(wèn)題.2014-07-07Python 使用PIL.Image制作運(yùn)動(dòng)小人的動(dòng)態(tài)圖思路詳解
這篇文章主要介紹了Python 使用PIL.Image制作一個(gè)運(yùn)動(dòng)小人的動(dòng)態(tài)圖,制作過(guò)程也很簡(jiǎn)單,只需要把圖片拆分成12等分,每幀大?。?7x165;連續(xù)讀取和播放就會(huì)形成動(dòng)態(tài)圖像,需要的朋友可以參考下2021-10-10