首页 > 解决方案 > 为什么 Python for 循环在遍历列表副本并进行删除时会跳过元素?

问题描述

我正在遍历一个列表,有时我会从另一个列表中删除一个元素。for 循环不会从另一个列表中删除元素,而是跳过整个实际迭代。我找到了解决方案,但我不太了解它。

这是一个代码示例。除了我注释掉了这一行之外,它是相同的两倍final_dt_list.remove(dt)

dt_list = [3600, 2700, 1800, 900]
final_dt_list = dt_list
for dt in dt_list:
    print(dt)
    if not((3600/dt).is_integer()):
        print('skipped '+str(dt))
        final_dt_list.remove(dt)
        continue
    print('ok')

print('\n \n')

dt_list = [3600, 2700, 1800, 900]
final_dt_list = dt_list
for dt in dt_list:
    print(dt)
    if not((3600/dt).is_integer()):
        print('skipped '+str(dt))
        #final_dt_list.remove(dt)
        continue
    print('ok')

我得到的结果是:

3600
ok
2700
skipped 2700
900
ok



3600
ok
2700
skipped 2700
1800
ok
900
ok

正如您在第一个循环的结果中看到的那样,没有打印 1800,只是没有任何反应。

最后对我有用的是创建一个我想要删除的项目列表。而不是创建一个新的循环来遍历我想要删除的项目。然后,这个新循环将从 final_dt_list 中删除每个元素。

在此处查看解决方案代码:

for r in removed:
    final_dt_list.remove(r)

为什么现在有效,而不是上面的例子?我很困惑。

提前致谢!

标签: pythonlistloopsfor-loopalias

解决方案


dt_listfinal_dt_list指向同一个对象(别名)。您在循环时编辑列表。由于您只保留用于循环的指针,因此您会记住位置,但由于元素消失,它们会发生变化。

尝试通过以下方式记住要删除的元素:

    dt_list = [3600, 2700, 1800, 900]
    rm_list = []
    for dt in dt_list:
        print(dt)
        if not((3600/dt).is_integer()):
            print('skipped '+str(dt))
            rm_list.append(dt)
            continue
        print('ok')
    dt_list = [x for x in dt_list if x not in rm_list]

编辑:

从一开始就使用列表理解会导致:

dt_list = [3600, 2700, 1800, 900]
dt_list = [x for x in dt_list if (3600/x).is_integer()]

推荐阅读