python 的列表遍歷刪除實(shí)現(xiàn)代碼
python的列表list可以用for循環(huán)進(jìn)行遍歷,實(shí)際開(kāi)發(fā)中發(fā)現(xiàn)一個(gè)問(wèn)題,就是遍歷的時(shí)候刪除會(huì)出錯(cuò),例如
l = [1,2,3,4] for i in l: if i != 4: l.remove(i) print l
這幾句話本來(lái)意圖是想清空列表l,只留元素4,但是實(shí)際跑起來(lái)并不是那個(gè)結(jié)果。再看下面,利用index來(lái)遍歷刪除列表l
l = [1, 2, 3, 4] for i in range(len(l)): if l[i] == 4: del l[i] print l
這樣沒(méi)問(wèn)題,可以遍歷刪除,但是列表l如果變?yōu)?l = [1,2,3,4,5]
如果還是按照上面的方法,設(shè)想一下,range開(kāi)始的范圍是0-4,中間遍歷的時(shí)候刪除了一個(gè)元素4,這個(gè)時(shí)候列表變成了= [1,2,3,5],這時(shí)候就會(huì)報(bào)錯(cuò)了,提示下標(biāo)超出了數(shù)組的表示,原因就是上面說(shuō)的遍歷的時(shí)候刪除了元素
所以python的list在遍歷的時(shí)候刪除元素一定要小心
可以使用filter過(guò)濾返回新的list
l = [1,2,3,4] l = filter(lambda x:x !=4,l) print l
這樣可以安全刪除l中值為4的元素了,filter要求兩個(gè)參數(shù),第一個(gè)是規(guī)則函數(shù),第二個(gè)參數(shù)要求輸入序列,而lambda這個(gè)函數(shù)的作用就是產(chǎn)生一個(gè)函數(shù),是一種緊湊小函數(shù)的寫法,一般簡(jiǎn)單的函數(shù)可以這么些
或者可以這樣
l = [1,2,3,4] l = [ i for i in l if i !=4]//同樣產(chǎn)生一個(gè)新序列,復(fù)值給l print l
或者干脆建立新的list存放要?jiǎng)h除的元素
l = [1,2,3,4] dellist = [] for i in l: if i == 4: dellist.append(i) for i in dellist: l.remove(i)
這樣也能安全刪除元素
所以要遍歷的時(shí)候刪除元素一定要小心,特別是有些操作并不報(bào)錯(cuò),但卻沒(méi)有達(dá)到預(yù)期的效果
上面說(shuō)到產(chǎn)生新序列,賦值等等,用python的id()這個(gè)內(nèi)置函數(shù)來(lái)看對(duì)象的id,可以理解為內(nèi)存中的地址,所以有個(gè)簡(jiǎn)要說(shuō)明
如果
l = [1,2,3,4] ll = l l.remove(1) print l//肯定是[2,3,4] print ll//這里會(huì)是什么?
如果用id函數(shù)查看的話就發(fā)現(xiàn)
print id(l),id(ll)
打印出相同的號(hào)碼,說(shuō)明他們其實(shí)是一個(gè)值,也就是說(shuō)上面的print ll將和l打印的一樣,所以python有這種性質(zhì),用的時(shí)候注意一下就行了
一般python書(shū)中會(huì)講到這個(gè)問(wèn)題
遍歷的時(shí)候如果想更新列表本身
推薦的做法是使用切片
l = [1, 2, 3, 4]
for i in l[:]
some code
python中循環(huán)刪除列表中元素時(shí)的坑!
循環(huán)刪除列表中元素時(shí)千萬(wàn)別用正序遍歷,一定要用反序遍歷!
廢話不多說(shuō),先上案例代碼:
def test(data): for i in data: data.remove(i) return data data = [1, 2, 3] print(test(data))
面對(duì)以上代碼,乍一看以為會(huì)打印出空列表,因?yàn)閠est函數(shù)內(nèi)通過(guò)for的方法將data中的元素都刪除了,其實(shí)不然,實(shí)際輸出如下:
[2]
為什么會(huì)產(chǎn)生這種結(jié)果呢? 我們來(lái)深度剖析一下:
原列表在內(nèi)存中為:
第一次執(zhí)行到data.remove(i)時(shí)將第一個(gè)元素‘1'刪除,列表變?yōu)椋?/p>
第二次執(zhí)行到data.remove(i)時(shí)i為第二個(gè)元素,即‘3',此時(shí)將‘3'刪除,列表變?yōu)椋?/p>
此時(shí)列表已經(jīng)沒(méi)有第三個(gè)元素了,即退出循環(huán),將[2]返回。
如何解決這個(gè)問(wèn)題呢? 我們可以用倒序刪除的方法!
直接上代碼:
def test(data): for i in data[::-1]: data.remove(i) return data data = [1, 2, 3] print(test(data))
此時(shí)再運(yùn)行,發(fā)現(xiàn)輸出結(jié)果為空列表
[]
倒序刪除原理如下:
原列表在內(nèi)存中為:
第一次執(zhí)行到data.remove(i)時(shí)是將倒數(shù)第一個(gè)元素‘3'刪除,列表變?yōu)椋?/p>
第二次執(zhí)行到data.remove(i)時(shí)i為倒數(shù)第二個(gè)元素,即‘2',此時(shí)將‘2'刪除,列表變?yōu)椋?/p>
第三次執(zhí)行到data.remove(i)時(shí)i為倒數(shù)第三個(gè)元素,即‘1',此時(shí)將‘1'刪除,列表變?yōu)榭樟斜恚?/p>
總結(jié):正序刪除列表中元素時(shí),被刪元素后面的值會(huì)向前頂,然后導(dǎo)致漏刪。倒序刪除元素時(shí),被刪元素前面的值不會(huì)向后靠,所以可以完整的遍歷到列表中所有的元素。
相關(guān)文章
python tkinter實(shí)現(xiàn)彩球碰撞屏保
這篇文章主要為大家詳細(xì)介紹了python tkinter實(shí)現(xiàn)彩球碰撞屏保,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-07-07python?使用ctypes調(diào)用C/C++?dll詳情
這篇文章主要介紹了python?使用ctypes調(diào)用C/C++?dll詳情,文章首先通過(guò)導(dǎo)入ctypes模塊,加載C/C++?dll到python進(jìn)程空間展開(kāi)主題相關(guān)內(nèi)容,需要的小伙伴可以參考一下2022-04-04PyQt5中多線程模塊QThread使用方法的實(shí)現(xiàn)
這篇文章主要介紹了PyQt5中多線程模塊QThread使用方法的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-01-01python實(shí)現(xiàn)人臉識(shí)別代碼
這篇文章主要介紹了python實(shí)現(xiàn)人臉識(shí)別代碼,還是比較不錯(cuò)的,這里分享個(gè)大家,共需要的朋友參考。2017-11-11Python可視化單詞統(tǒng)計(jì)詞頻統(tǒng)計(jì)中文分詞的實(shí)現(xiàn)步驟
這篇文章主要介紹了Python可視化單詞統(tǒng)計(jì)詞頻統(tǒng)計(jì)中文分詞,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-11-11Python網(wǎng)絡(luò)編程之Socket和HTTP協(xié)議詳解
在當(dāng)今互聯(lián)網(wǎng)時(shí)代,網(wǎng)絡(luò)編程是程序員不可或缺的一項(xiàng)技能,本文將介紹Python中兩種主要的網(wǎng)絡(luò)編程方式:Socket編程和基于HTTP協(xié)議的網(wǎng)絡(luò)編程,希望對(duì)大家有所幫助2024-04-04