11

我是 Python 新手,正在尝试使用 Python 3.2.3(默认,2012 年 10 月 19 日,20:13:42)的列表,Linux2 上的 [GCC 4.6.3]

这是我的示例代码

>>> l=[1,2,3,4,5,6]
>>> for i in l:
...     l.pop(0)
...     print(l)
... 

我希望以下输出

1
[2, 3, 4, 5, 6]
2
[3, 4, 5, 6]
3
[4, 5, 6]
4
[5, 6]
5
[6]
6
[]

相反,我得到了这个

1
[2, 3, 4, 5, 6]
2
[3, 4, 5, 6]
3
[4, 5, 6]

for 循环在 3 圈后停止迭代。有人可以解释为什么吗?

4

6 回答 6

13

展开一点(插入符号 ( ^) 位于循环“索引”处):

your_list = [1,2,3,4,5,6]
             ^

弹出第一个项目后:

your_list = [2,3,4,5,6]
             ^

现在继续循环:

your_list = [2,3,4,5,6]
               ^

现在弹出第一项:

your_list = [3,4,5,6]
               ^

现在继续循环:

your_list = [3,4,5,6]
                 ^

现在弹出第一项:

your_list = [4,5,6]
                 ^

现在继续循环——等等,我们完成了。:-)


>>> l = [1,2,3,4,5,6]
>>> for x in l:
...     l.pop(0)
... 
1
2
3
>>> print l
[4, 5, 6]
于 2012-12-18T18:44:26.347 回答
11

您可以为此任务使用while循环而不是循环。for

while len(some_list)>0 :
    some_list.pop(0)

循环实际上会遍历列表中的for每个项目,这将不起作用,因为列表中的索引会随着每次删除而改变,并且您最终不会得到所有项目。

但是,while每次循环运行时,循环都会检查一个条件,如果仍然为真,则再次运行代码。这里我们指定列表的长度必须大于0,即列表中必须有内容。

于 2017-10-06T21:08:15.240 回答
3

尝试修改要迭代的集合时必须小心。在这种情况下,列表使用简单的整数索引来跟踪“当前位置”。当您使用 时pop(),所有内容都会更改索引,因此会跳过元素。

在循环的第一次迭代中,i 是 l[0]。然后弹出列表,然后访问 l[1],它最初位于 l[2]。然后你弹出列表,下一次迭代访问 l[2],这就是以前在 l[4] 的位置,等等。

无论如何都不需要在这段代码中弹出元素,大概你在你的真实代码中做一些更复杂的事情。

于 2012-12-18T18:40:37.603 回答
2

Python 不支持在迭代列表时更改列表的长度。使用副本或使用列表推导代替。

想想 Python 实际上是如何执行 for 循环的——它通过元素进行计数,返回当前索引处的项目。当您删除一个时,索引意味着不同的元素。

于 2012-12-18T18:40:20.703 回答
1

代码

l = [1,2,3,4,5,6]
for i in range(len(l)):
    l.pop(0)
    print(l)

返回

[2, 3, 4, 5, 6]
[3, 4, 5, 6]
[4, 5, 6]
[5, 6]
[6]
[]
于 2020-07-16T02:06:58.550 回答
1
li = [1, 2, 3, 4, 5]

while li:
    # For descending order
    # print(li.pop())

    # For ascending order
    print(li.pop(0))
于 2022-01-05T18:52:45.997 回答