为什么当我迭代列表并修改它时 Python 会跳过元素?

2025-02-05 13:24:00
admin
原创
66
摘要:问题描述:我目前正在用 Python 开发一个程序,我刚刚注意到语言中的 foreach 循环或列表结构有问题。为了简化起见,我仅给出一个问题的通用示例,因为我的程序和通用示例都出现了相同的错误行为:x = [1,2,2,2,2] for i in x: x.remove(i) print x ...

问题描述:

我目前正在用 Python 开发一个程序,我刚刚注意到语言中的 foreach 循环或列表结构有问题。为了简化起见,我仅给出一个问题的通用示例,因为我的程序和通用示例都出现了相同的错误行为:

x = [1,2,2,2,2]

for i in x:
    x.remove(i)

print x        

好吧,这里的问题很简单,我以为这段代码应该从列表中删除所有元素。问题是,执行后,我总是得到列表中剩余的 2 个元素。

我做错什么了?

我并不是真的想清空列表,这只是一个例子......


解决方案 1:

这是 Python 中一个有据可查的行为,即您不应该修改正在迭代的列表。请尝试以下方法:

for i in x[:]:
    x.remove(i)

返回[:]的“切片” x,它恰好包含其所有元素,因此实际上是 的副本x

解决方案 2:

当您删除一个元素时,for 循环会转到下一个索引,然后您就会跳过一个元素。

反过来做。或者请陈述你的真正问题。

解决方案 3:

我认为,从广义上讲,当你写:

for x in lst:
    # loop body goes here

在底层,python 正在做这样的事情:

i = 0
while i < len(lst):
    x = lst[i]
    # loop body goes here
    i += 1

如果您插入lst.remove(x)循环体,也许您就能明白为什么会得到这样的结果?

本质上,python 使用移动指针来遍历列表。指针首先指向第一个元素。然后删除第一个元素,从而使第二个元素成为新的第一个元素。然后指针移动到新的第二个(之前是第三个)元素。依此类推。(如果您使用 [1,2,3,4,5] 而不是 [1,2,2,2,2] 作为示例列表,可能会更清楚)

解决方案 4:

你为什么不直接使用:

x = []

这可能是因为您正在更改您正在迭代的同一数组。

如果您想以自己的方式清除阵列,请尝试 Chris-Jester Young 的回答。

解决方案 5:

我知道这是一篇有可接受答案的旧帖子,但对于那些可能仍会出现的人来说......

之前的一些答案表明,在迭代过程中更改可迭代对象是个坏主意。但作为一种强调正在发生的事情的方式...

>>> x=[1,2,3,4,5]
>>> for i in x:
...     print i, x.index(i)
...     x.remove(i)
...     print x
...
1 0
[2, 3, 4, 5]
3 1
[2, 4, 5]
5 2
[2, 4]

希望视觉效果能够帮助澄清。

解决方案 6:

我同意 John Fouhy 关于中断条件的观点。正如 Chris Jester-Young 所建议的那样,遍历列表的副本对于 remove() 方法有效。但如果需要 pop() 特定项,那么反向迭代就可以了,正如 Erik 所提到的,在这种情况下可以就地完成操作。例如:

def r_enumerate(iterable):
    """enumerator for reverse iteration of an iterable"""
    enum = enumerate(reversed(iterable))
    last = len(iterable)-1
    return ((last - i, x) for i,x in enum)

x = [1,2,3,4,5]
y = []
for i,v in r_enumerate(x):
    if v != 3:
        y.append(x.pop(i))
    print 'i=%d, v=%d, x=%s, y=%s' %(i,v,x,y)

或者使用 xrange:

x = [1,2,3,4,5]
y = []
for i in xrange(len(x)-1,-1,-1):
    if x[i] != 3:
        y.append(x.pop(i))
    print 'i=%d, x=%s, y=%s' %(i,x,y)

解决方案 7:

如果您需要从列表中过滤内容,使用列表推导可能是一个更好的主意:

newlist = [x for x in oldlist if x%2]

例如,从整数列表中过滤掉所有偶数

解决方案 8:

存储在计算机内存中的列表。这处理指向内存工件的指针。当您在逐元素循环中删除一个元素时,您会将指针移动到内存地址中的下一个可用元素

您正在修改内存并对其进行迭代。指向元素的指针会穿过列表移动到下一个可用位置。因此,如果大小为 5...请在此处输入代码

 [**0**,1,2,3,4]
remove 0   --->  [1,**2**,3,4]  pointer moves to second index.
remove 2   --->  [1,3,**4**] pointer moves to 3rd index.
remove 4   --->  [1,3]

当我的学生使用 pop(1) 时,我刚刚向他们解释了这一点。另一个非常有趣的副作用错误。

x=[1,**2**,3,4,5]
for i in x:
  x.pop(1)
  print(x,i)

[1, **3**, 4, 5] 1   at index 0 it removed the index 1 (2)
[1, **4**, 5] 3      at index 1 it removed the index 1 (3)
[1, 5] 5         at index 2 it removed the index 1 (4)

呵呵。他们就像为什么这不起作用......我的意思是......它确实......完全按照你告诉它的方式工作。不是读心术。:)

相关推荐
  政府信创国产化的10大政策解读一、信创国产化的背景与意义信创国产化,即信息技术应用创新国产化,是当前中国信息技术领域的一个重要发展方向。其核心在于通过自主研发和创新,实现信息技术应用的自主可控,减少对外部技术的依赖,并规避潜在的技术制裁和风险。随着全球信息技术竞争的加剧,以及某些国家对中国在科技领域的打压,信创国产化显...
工程项目管理   1565  
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   1354  
  信创国产芯片作为信息技术创新的核心领域,对于推动国家自主可控生态建设具有至关重要的意义。在全球科技竞争日益激烈的背景下,实现信息技术的自主可控,摆脱对国外技术的依赖,已成为保障国家信息安全和产业可持续发展的关键。国产芯片作为信创产业的基石,其发展水平直接影响着整个信创生态的构建与完善。通过不断提升国产芯片的技术实力、产...
国产信创系统   21  
  信创生态建设旨在实现信息技术领域的自主创新和安全可控,涵盖了从硬件到软件的全产业链。随着数字化转型的加速,信创生态建设的重要性日益凸显,它不仅关乎国家的信息安全,更是推动产业升级和经济高质量发展的关键力量。然而,在推进信创生态建设的过程中,面临着诸多复杂且严峻的挑战,需要深入剖析并寻找切实可行的解决方案。技术创新难题技...
信创操作系统   27  
  信创产业作为国家信息技术创新发展的重要领域,对于保障国家信息安全、推动产业升级具有关键意义。而国产芯片作为信创产业的核心基石,其研发进展备受关注。在信创国产芯片的研发征程中,面临着诸多复杂且艰巨的难点,这些难点犹如一道道关卡,阻碍着国产芯片的快速发展。然而,科研人员和相关企业并未退缩,积极探索并提出了一系列切实可行的解...
国产化替代产品目录   28  
热门文章
项目管理软件有哪些?
云禅道AD
禅道项目管理软件

云端的项目管理软件

尊享禅道项目软件收费版功能

无需维护,随时随地协同办公

内置subversion和git源码管理

每天备份,随时转为私有部署

免费试用