从列表中删除项目时出现奇怪的结果

Fin*_*ist 32 python loops list

我有这段代码:

numbers = range(1, 50)

for i in numbers:
    if i < 20:
        numbers.remove(i)

print(numbers)
Run Code Online (Sandbox Code Playgroud)

但我得到的结果是:

[2,4,6,8,10,12,14,16,18,20,21,22,23,24,25,26,27,28,29,30,31,32,33,34,35 ,36,37,38,39,40,41,42,43,44,45,46,47,48,49]

当然我希望结果中没有出现20以下的数字,我假设我在删除时做错了.

sen*_*rle 65

您在迭代时修改列表.这意味着第一次通过循环i == 1,所以1从列表中删除.然后for循环转到列表中的第二个项目,它不是2,而是3!然后将其从列表中删除,然后for循环继续到列表中的第三个项目,现在是5.依此类推.也许这样可视化更容易,^指向以下值i:

[1, 2, 3, 4, 5, 6...]
 ^
Run Code Online (Sandbox Code Playgroud)

这是最初列表的状态; 然后删除1,循环转到列表中的第二项:

[2, 3, 4, 5, 6...]
    ^
[2, 4, 5, 6...]
       ^
Run Code Online (Sandbox Code Playgroud)

等等.

迭代它时没有好的方法来改变列表的长度.你能做的最好的事情是这样的:

numbers = [n for n in numbers if n >= 20]
Run Code Online (Sandbox Code Playgroud)

或者,对于就地更改(parens中的东西是生成器表达式,在切片赋值之前隐式转换为元组):

numbers[:] = (n for in in numbers if n >= 20)
Run Code Online (Sandbox Code Playgroud)

如果你想在删除它之前对n执行操作,你可以尝试的一个技巧是:

for i, n in enumerate(numbers):
    if n < 20 :
        print "do something" 
        numbers[i] = None
numbers = [n for n in numbers if n is not None]
Run Code Online (Sandbox Code Playgroud)

  • 关于“for”从Python文档中保留索引的相关注释https://docs.python.org/3.9/reference/compound_stmts.html#the-for-statement:“*当序列被修改时有一个微妙之处循环(这只能发生在可变序列中,例如列表)。内部计数器用于跟踪接下来使用哪个项目,并且在每次迭代时都会递增。...这意味着如果套件删除当前的项目(或序列中的前一个)项目,下一个项目将被跳过(因为它获取已被处理的当前项目的索引)。*” (3认同)

eyq*_*uem 7

从列表中删除项目很简单:从列表的末尾开始:

li = range(1,15)
print li,'\n'

for i in xrange(len(li)-1,-1,-1):
    if li[i] < 6:
        del li[i]

print li
Run Code Online (Sandbox Code Playgroud)

结果

[1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14] 

[6, 7, 8, 9, 10, 11, 12, 13, 14]
Run Code Online (Sandbox Code Playgroud)

  • 我仍然不明白你的评论,因为列表是否被打乱或排序并不重要,这段代码仍然有效。 (3认同)
  • @mikerodent 不,不是。当您想要在迭代列表的同时修改列表时,向后操作是很常见的 (2认同)

Tru*_*ufa 5

@remitle的答案是要走的路!

话虽如此,为了进一步说明你的问题,如果你考虑一下,你总是想要删除索引0二十次:

[1,2,3,4,5............50]
 ^
[2,3,4,5............50]
 ^
[3,4,5............50]
 ^
Run Code Online (Sandbox Code Playgroud)

所以你真的可以这样做:

aList = range(50)
i = 0
while i < 20:
    aList.pop(0)
    i += 1

print aList #[21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49]
Run Code Online (Sandbox Code Playgroud)

我希望它有所帮助.


以下提供的是坏的做法,据我所知.

编辑(更多):

lis = range(50)
lis = lis[20:]
Run Code Online (Sandbox Code Playgroud)

也会做这个工作.

EDIT2(我很无聊):

functional = filter(lambda x: x> 20, range(50))
Run Code Online (Sandbox Code Playgroud)