在 Python 列表和字典常踩的一个坑

本文介绍了在Python中遍历列表和字典时遇到的修改数据问题及其解决方法。对于列表,通过示例揭示了在遍历过程中移除元素导致的错误,提出两种解决方案:翻转列表或复制后再遍历。对于字典,解释了不能直接遍历修改的原因,并给出先复制键再迭代删除的解决策略。
部署运行你感兴趣的模型镜像

【导语】:在Python中,如果我们想要在遍历一组数据的过程中,对这组数据进行修改,通常会出现许多问题,例如对列表进行上述操作时, 会忽略部分数据;遍历字典时,不能修改数据。本文针对这些问题,提出了多种解决方案。

简介

一、关于列表

1.问题描述

在Python中,如果你试图在遍历一组数据的过程中,对其进行修改,这通常没什么问题。例如:


l = [3, 4, 56, 7, 10, 9, 6, 5]

for i in l:
    if not i % 2 == 0:
        continue
    l.remove(i)

print(l)

上述这段代码遍历了一个包含数字的列表,为了去除掉所有偶数,直接修改了列表l。然而,运行后输出却是:

[3, 56, 7, 9, 5]

等一下!输出似乎不对。最终的结果仍然含有一个偶数56。为什么没有成功去除这个数呢?我们可以尝试打印出 for循环遍历的所有元素,运行如下代码:

l = [3, 4, 56, 7, 10, 9, 6, 5]

for i in l:
    print(i)
    if not i % 2 == 0:
        continue
    l.remove(i)

print(l)

这段代码的输出为:

3
4
7
10
6
[3, 56, 7, 9, 5]

从输出可以看出,for循环似乎没有访问列表中的所有元素。为了解for循环在内部究竟做了什么, 我们可以使用 iter 和 next 来模拟一下。看看下面这个例子,我使用了ipython shell 来运行代码:


In [1]: l = [3, 4, 56, 7, 10, 9, 6, 5]

In [2]: # 把列表变成一个迭代器

In [3]: it = iter(l)

In [4]: # 使用 next() 方法来模拟 for循环

In [5]: next(it)
Out[5]: 3

In [6]: next(it)
Out[6]: 4

In [7]: # 移除一个迭代器已经访问过的元素

In [8]: l.remove(3)

In [9]: next(it)
Out[9]: 7

In [10]: # 注意此处跳过了56,我们可以再移除一个元素

In [11]: l.remove(4)

In [12]: next(it)
Out[12]: 9

上面这个实验揭示了:当你移除一个迭代器已经访问过的元素后,在下一次迭代时,会跳过右边的一个元素,直接访问下一个。

反之依然成立,即当开始迭代后,如果你在列表开头添加了一个元素,下次迭代时,可能会访问到已经迭代过的元素,下面这段代码就出现了这种情况:

In[1]: l = [3, 4, 56, 7, 10, 9, 6, 5]

In[2]: it = iter(l)

In[3]: next(it)
Out[3]: 3

In[4]: next(it)
Out[4]: 4

In[5]: l.insert(0, 44)

In[6]: next(it)
Out[6]: 4

注意当在列表头部添加了44后,4被访问了两次。

2.解决方案

为了解决上述问题,我们必须得确保:不能移除迭代器访问过的元素。

方案一

我们可以先对原列表进行翻转得到一个新列表,再对新列表进行迭代,并在原列表 l 中移除不符合条件的元素。该方案代码如下:

l = [3, 4, 56, 7, 10, 9, 6, 5]

# 迭代翻转后的列表
for i in reversed(l):
    print(i)
    if not i % 2 == 0:
        continue
    l.remove(i)

print(l)

结果如下:

5
6
9
10
7
56
4
3
[3, 7, 9, 5]

注意,迭代器现在成功访问到了列表中的所有元素,并最终输出了只含有奇数的列表。

方案二

我们还可以在开始迭代前,先复制列表 l 。但是当列表 l 中的数据过多时,这样做显然比较耗费性能。该方案代码如下:

l = [3, 4, 56, 7, 10, 9, 6, 5]

# 在这里使用 'l.copy()' 来对列表 l 进行浅拷贝
for i in l.copy():  
    print(i)   
    if not i % 2 == 0:     
        continue  
    l.remove(i)
    
print(l)

输出如下:

3
4
56
7
10
9
6
5
[3, 7, 9, 5]

该方案能保证迭代的顺序和移除元素的顺序相同。不过由于迭代和移除这两种操作针对的是两个不同的列表,因此顺序相同并不重要。

二、关于字典

1.问题描述

在对字典进行迭代时,不能修改字典。如下:

# {0: 0, 1: 1, 2: 2, 3: 3, 4: 4, 5: 5, 6: 6, 7: 7, 8: 8, 9: 9}
d = {k: k for k in range(10)}

for k, v in d.items():  
    if not v % 2 == 0:    
        continue  
    d.pop(k)

这段代码会产生 RuntimeError :

Traceback (most recent call last):  
  File "F:/Documents/pythonprojects/01practice/app.py", line 7, in <module>  
    for k, v in d.items():
RuntimeError: dictionary changed size during iteration

2.解决方案

我们可以先复制字典的所有 key ,随后在迭代 key 的过程中,移除不符合条件的元素。过程如下:

# {0: 0, 1: 1, 2: 2, 3: 3, 4: 4, 5: 5, 6: 6, 7: 7, 8: 8, 9: 9}
d = {k: k for k in range(10)}
# 这里复制了字典中的所有key值
# 没有复制整个字典
# 同时使用tuple()速度更快
for k in tuple(d.keys()):   
    if not d[k] % 2 == 0:    
        continue  
    d.pop(k)
    
print(d)

运行代码后输出如下:


{1: 1, 3: 3, 5: 5, 7: 7, 9: 9}

我们成功移除了字典中的所有偶数键值对!

结论

在本文中我们针对迭代一组数据时无法进行修改的问题,分别提出了不同的解决方案:如果想在遍历列表的时候,对列表进行修改, 我们可以先对原列表进行翻转或复制,从而得到一个新列表,随后在遍历新列表的过程中,修改原列表中的数据;如果我们想在遍历字典的时候,对字典进行修改,可以先复制字典的所有键值,然后在迭代键值的时候,修改字典中的数据。


 

您可能感兴趣的与本文相关的镜像

Python3.8

Python3.8

Conda
Python

Python 是一种高级、解释型、通用的编程语言,以其简洁易读的语法而闻名,适用于广泛的应用,包括Web开发、数据分析、人工智能和自动化脚本

<think>嗯,用户问的是Python 3Python有什么区别。我需要先确认自己对这个问题的理解是否正确。Python指的是Python 2.x版本,而Python 3是后续的重大更新,两者不兼容。用户可能刚开始学习Python,遇到了版本问题,或者在项目中需要选择版本,所以想知道它们的区别应该用哪个。 首先,我应该回顾Python 3Python 2的主要区别。比如print函数在Python 3中是print(),而在Python 2中是print语句。还有整数除法,Python 3里3/2得到1.5,而Python 2里是1。Unicode支持也是一个重点,Python 3默认字符串是Unicode,而Python 2需要前缀u。此外,语法上的变化,比如异处理、xrangerange的区别,以及一些标准库的模块变化,如urllib。 然后要考虑用户可能的背景。可能是新手,对版本差异感到困惑,或者需要迁移旧代码。需要解释清楚两者的不兼容性,以及为什么Python 3现在成为主流。要强调Python 2已经停止维护,所以新项目应该用Python 3。 可能用户没有提到的深层需求是,如何选择版本,或者如何处理不同版本带来的问题。因此,回答中不仅要列出区别,还要给出建议,比如使用Python 3.6之后的版本,使用工具进行迁移等。 需要避免使用过于技术性的术语,保持口语化,但关键点要清晰。比如解释print的不同时,给出例子,说明语法变化。同时,提醒用户注意编码问题,比如处理中文时的不同表现,避免乱码。 还要检查是否有遗漏的重要区别,比如性能改进、新特性(如类型注解、异步编程),这些也是Python 3的优势。此外,社区支持第三方库的兼容性,现在大多数库已经支持Python 3,所以不用担心生态问题。 最后,总结时强调使用Python 3的必要性,并给出学习建议,如安装最新版本,注意教程的版本等。确保回答结构清晰,先讲主要区别,再讲其他改进,最后给建议,这样用户能一步步理解。</think>Python 3 Python 2 是 Python 语言的两个主要版本,它们之间存在一些**不兼容的语法差异功能改进**。以下是主要区别核心总结: --- ### 1. **核心区别** - **发布时间**: - Python 2:2000 年发布,2020 年已停止官方支持。 - Python 3:2008 年发布,是当前主流版本,持续更新(如 Python 3.11+)。 - **兼容性**: - Python 3 **不向后兼容** Python 2,导致旧代码可能需要修改才能运行。 --- ### 2. **语法差异** - **`print` 函数**: - Python 2:`print "Hello"`(语句) - Python 3:`print("Hello")`(函数) - **整数除法**: - Python 2:`3 / 2 = 1`(自动取整),Python 3:`3 / 2 = 1.5` - Python 2 需用 `3 // 2` 显式取整。 - **字符串与编码**: - Python 2:默认字符串是 `str`(字节序列),需用 `u"中文"` 表示 Unicode。 - Python 3:默认字符串是 Unicode,`bytes` 类型表示二进制数据(如 `b"text"`)。 - **`xrange` 与 `range`**: - Python 2:`range()` 生成列表,`xrange()` 生成迭代器。 - Python 3:移除 `xrange`,`range()` 直接生成迭代器。 --- ### 3. **功能改进** - **类型注解**: - Python 3.5+ 支持类型提示(如 `def func(a: int) -> str`),方便静态检查。 - **异步编程**: - Python 3.5+ 引入 `async/await` 语法,简化协程开发。 - **性能优化**: - Python 3 在内存管理、字典实现等方面性能更好。 --- ### 4. **其他差异** - **异捕获**: - Python 3:`except Exception as e`(必须用 `as`)。 - Python 2:允许 `except Exception, e`。 - **输入函数**: - Python 3:`input()` 直接返回字符串。 - Python 2:`input()` 会执行表达式,推荐用 `raw_input()`。 --- ### **如何选择?** - **新项目必选 Python 3**:Python 2 已淘汰,社区第三方库(如 NumPy、Django)全面转向 Python 3。 - **旧项目迁移工具**:可用 `2to3` 工具自动转换代码。 --- ### 示例对比 ```python # Python 2 print "Hello" print 3 / 2 # 输出 1 text = u"中文" # Python 3 print("Hello") print(3 / 2) # 输出 1.5 text = "中文" # 默认支持 Unicode ``` -- 如果初学 Python,**直接选择 Python 3.6 及以上版本**即可(如 3.11),避免旧版兼容的“”。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值