Python陷阱-如何安全地刪除列表元素?
一個常見的任務(wù)是在一個列表上迭代,并根據(jù)條件刪除一些元素。本文將展示如何完成該任務(wù)的不同方法,同時展示一些需要避免的陷阱。
假設(shè)我們需要修改列表a,并且必須刪除所有不是偶數(shù)的項。首先實現(xiàn)輔助函數(shù)even(x)來確定一個數(shù)字x是否是偶數(shù):
a = [1, 2, 2, 3, 4]
def even(x):
return x % 2 == 0
方法1: 創(chuàng)建新列表,過濾元素
1a) 列表推導(dǎo),創(chuàng)建新列表
使用列表推導(dǎo)創(chuàng)建一個新的列表,只包含你不想刪除的元素,并把它分配回a:
a = [1, 2, 2, 3, 4]
def even(x):
return x % 2 == 0
# 列表推導(dǎo),但創(chuàng)建了一個新的變量a
a = [x for x in a if not even(x)]
# --> a = [1, 3]
print(a)
你可以在10個python小技巧文章中了解更多關(guān)于列表推導(dǎo)的知識。
1b) 列表推導(dǎo),對a[:]賦值
上面的代碼創(chuàng)建了一個新的變量a。我們也可以通過賦值給切片a[:]就地改變現(xiàn)有的列表。這種方法更有效率,如果有其他對a的引用需要反映變化的話,這種方法可能很有用。
a = [1, 2, 2, 3, 4]
def even(x):
return x % 2 == 0
# 列表推導(dǎo),但賦值給a[:] 就地改變列表
a[:] = [x for x in a if not even(x)]
# --> a = [1, 3]
print(a)
1c) 使用itertools.filterfalse()
itertools模塊為非常有效的循環(huán)迭代提供了各種函數(shù),并且提供了一種過濾元素的方法。
a = [1, 2, 2, 3, 4]
def even(x):
return x % 2 == 0
# 通過itertools 快速過濾
from itertools import filterfalse
a[:] = filterfalse(even, a)
# --> a = [1, 3]
print(a)
方法2:列表副本上迭代
如果你真的想保留for語法,那么需要在列表的副本上進(jìn)行迭代(副本可以通過使用a[:]簡單創(chuàng)建)?,F(xiàn)在你可以在條件為True時從原始列表中刪除元素:
a = [1, 2, 2, 3, 4]
def even(x):
return x % 2 == 0
# 注意是在列表副本a[:] 上循環(huán)
for item in a[:]:
if even(item):
a.remove(item)
# --> a = [1, 3]
print(a)
常見陷阱
千萬別在同一個列表上循環(huán),并在迭代過程中修改它!
這和上面的代碼是一樣的,只是沒有在副本上循環(huán)。刪除一個元素將使所有后續(xù)元素向左移動一個位置,因此在下一次迭代中,一個元素將被跳過。這可能會導(dǎo)致不正確的結(jié)果:
a = [1, 2, 2, 3, 4]
def even(x):
return x % 2 == 0
# 直接在變量a上進(jìn)行循環(huán),沒有在副本上
for item in a:
if even(item):
a.remove(item)
# --> a = [1, 2, 3] !!!
print(a)
另外,在列表的循環(huán)過程中,千萬不要修改索引!
這是不正確的,因為在循環(huán)中改變i不會影響下一次迭代中i的值。這個例子也會產(chǎn)生非預(yù)期的效果,甚至?xí)?dǎo)致IndexErrors,比如這里:
a = [1, 2, 2, 3, 4]
def even(x):
return x % 2 == 0
# 試圖在循環(huán)在改變索引i,但出錯!
for i in range(len(a)):
if even(a[i]):
del a[i]
i -= 1
# --> IndexError: list index out of range
print(a)
小節(jié)
通過上面的學(xué)習(xí)相信你現(xiàn)在可以安心地刪除列表的部分元素.