日韩久久久精品,亚洲精品久久久久久久久久久,亚洲欧美一区二区三区国产精品 ,一区二区福利

Python中l(wèi)ist循環(huán)遍歷刪除數(shù)據(jù)的正確方法

系統(tǒng) 2364 0

前言

初學Python,遇到過這樣的問題,在遍歷list的時候,刪除符合條件的數(shù)據(jù),可是總是報異常,代碼如下:

            
num_list = [1, 2, 3, 4, 5]
print(num_list)

for i in range(len(num_list)):
 if num_list[i] == 2:
  num_list.pop(i)
 else:
  print(num_list[i])

print(num_list)
          

會報異常: IndexError: list index out of range

原因是在刪除list中的元素后,list的實際長度變小了,但是循環(huán)次數(shù)沒有減少,依然按照原來list的長度進行遍歷,所以會造成索引溢出。

于是我修改了代碼如下:

            
num_list = [1, 2, 3, 4, 5]
print(num_list)

for i in range(len(num_list)):
 if i >= len(num_list):
  break

 if num_list[i] == 2:
  num_list.pop(i)
 else:
  print(num_list[i])

print(num_list)
          

這回不會報異常了,但是打印結果如下:

[1, 2, 3, 4, 5]
1
4
5
[1, 3, 4, 5]
[Finished in 0.441s]

雖然最后,list中的元素[2]確實被刪除掉了,但是,在循環(huán)中的打印結果不對,少打印了[3]。

思考了下,知道了原因,當符合條件,刪除元素[2]之后,后面的元素全部往前移,于是[3, 4, 5]向前移動,那么元素[3]的索引,就變成了之前[2]的索引(現(xiàn)在[3]的下標索引變?yōu)?了),后面的元素以此類推。可是,下一次for循環(huán)的時候,是從下標索引2開始的,于是,取出了元素[4],就把[3]漏掉了。

把代碼修改成如下,結果一樣,絲毫沒有改觀:

            
num_list = [1, 2, 3, 4, 5]
print(num_list)

for item in num_list:
 if item == 2:
  num_list.remove(item)
 else:
  print(item)

print(num_list)
          

既然知道了問題的根本原因所在,想要找到正確的方法,也并不難,于是我寫了如下的代碼:

            
num_list = [1, 2, 3, 4, 5]
print(num_list)

i = 0
while i < len(num_list):
 if num_list[i] == 2:
  num_list.pop(i)
  i -= 1
 else:
  print(num_list[i])

 i += 1

print(num_list)
          

執(zhí)行結果,完全正確:

[1, 2, 3, 4, 5]
1
3
4
5
[1, 3, 4, 5]
[Finished in 0.536s]

我的做法是,既然用for循環(huán)不行,那就換個思路,用while循環(huán)來搞定。每次while循環(huán)的時候,都會去檢查list的長度(i < len(num_list)),這樣,就避免了索引溢出,然后,在符合條件,刪除元素[2]之后,手動把當前下標索引-1,以使下一次循環(huán)的時候,通過-1后的下標索引取出來的元素是[3],而不是略過[3]。

當然,這還不是最優(yōu)解,所以,我搜索到了通用的解決方案:1、倒序循環(huán)遍歷;2、遍歷拷貝的list,操作原始的list。

1、倒序循環(huán):

            
num_list = [1, 2, 3, 4, 5]
print(num_list)

for i in range(len(num_list)-1, -1, -1):
 if num_list[i] == 2:
  num_list.pop(i)
 else:
  print(num_list[i])

print(num_list)
          

執(zhí)行結果完全正確。那么,為何正序循環(huán)時刪除就有問題,而倒序循環(huán)時刪除就ok?額。。。。。。言語難表,還是畫個丑圖出來吧。

1)正序循環(huán)時刪除:

Python中l(wèi)ist循環(huán)遍歷刪除數(shù)據(jù)的正確方法_第1張圖片

刪除元素[2]之后,下一次循環(huán)的下標索引為2,但此時,里面存放的是[4],于是就把[3]給漏了。

2)倒序循環(huán)時刪除

Python中l(wèi)ist循環(huán)遍歷刪除數(shù)據(jù)的正確方法_第2張圖片

刪除元素[2]后,[3, 4, 5]往前擠,但是沒關系,因為下一次循環(huán)的下標索引為0,里面存放的是[1],所以正是我們所期望的正確的元素值。

2、遍歷拷貝的list,操作原始的list

            
num_list = [1, 2, 3, 4, 5]
print(num_list)

for item in num_list[:]:
 if item == 2:
  num_list.remove(item)
 else:
  print(item)

print(num_list)
          

原始的list是num_list,那么其實,num_list[:]是對原始的num_list的一個拷貝,是一個新的list,所以,我們遍歷新的list,而刪除原始的list中的元素,則既不會引起索引溢出,最后又能夠得到想要的最終結果。此方法的缺點可能是,對于過大的list,拷貝后可能很占內存。那么對于這種情況,可以用倒序遍歷的方法來實現(xiàn)。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對腳本之家的支持。


更多文章、技術交流、商務合作、聯(lián)系博主

微信掃碼或搜索:z360901061

微信掃一掃加我為好友

QQ號聯(lián)系: 360901061

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。

【本文對您有幫助就好】

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描上面二維碼支持博主2元、5元、10元、自定義金額等您想捐的金額吧,站長會非常 感謝您的哦!!!

發(fā)表我的評論
最新評論 總共0條評論
主站蜘蛛池模板: 乐亭县| 兴安盟| 苗栗县| 长宁县| 夏津县| 张家口市| 鄯善县| 抚顺市| 门源| 石阡县| 若尔盖县| 青神县| 枝江市| 东乡县| 望都县| 容城县| 全南县| 平乡县| 中山市| 金秀| 奉节县| 精河县| 黄大仙区| 罗山县| 石棉县| 岳普湖县| 河北省| 德兴市| 方城县| 桂阳县| 墨脱县| 汉阴县| 红桥区| 马山县| 上饶市| 观塘区| 林周县| 巫溪县| 塘沽区| 巴中市| 宝坻区|