0

0

Python 可迭代对象设计的工程思维

舞夢輝影

舞夢輝影

发布时间:2026-02-18 10:38:03

|

410人浏览过

|

来源于php中文网

原创

__iter__ 必须返回新迭代器而非 self,因可迭代对象与迭代器职责分离:前者负责生成,后者管理状态;否则多次遍历失败,违反pep 234契约。

python 可迭代对象设计的工程思维

为什么 __iter__ 返回迭代器,而不是直接返回 self

因为可迭代对象Iterable)和迭代器(Iterator)在语义和生命周期上必须分离。常见错误是让类同时实现 __iter____next__ 并在 __iter__ 里返回 self,这会导致多次遍历失败——比如在 for 循环、list()sum() 中重复使用同一个对象时,第二次调用就直接空了。

正确做法是:每次调用 __iter__ 都新建一个独立的迭代器实例。这个迭代器内部持有当前状态(如索引、游标),而可迭代对象本身只负责“生成”它。

  • 适用场景:rangedict.keys()、自定义数据容器(如树、链表)
  • 性能影响:新建对象开销极小,远小于状态错乱带来的逻辑 bug
  • 兼容性:符合 PEP 234,所有标准库函数(如 itertools.chain)都依赖这一契约

StopIteration 该由谁抛、什么时候抛

只能由迭代器的 __next__ 方法抛,且仅在“真的没有下一个元素”时抛。不能靠计数预判,也不能在初始化或 __iter__ 里抛。

常见错误包括:在 __next__ 开头就检查 if self.index >= len(self.data) 然后抛,但没处理空数据或边界变化;或者误把 IndexErrorStopIteration 捕获并吞掉,导致无限循环。

立即学习Python免费学习笔记(深入)”;

Taygod免费企业建站系统 A1.1
Taygod免费企业建站系统 A1.1

TAYGOD免费企业建站系统是一款开源的免费程序,您可以 TAYGOD免费企业建站系统ASP版是一款基于asp+access的免费开源建站系统。整套系统的设计构造,完全考虑中小企业类网站的功能要求,网站后台功能强大,管理简捷,支持模板机制,能够快速建立您的企业网站。 系统特性: 采用流行的asp+access设计,功能强,实用性高。 代码美工完全分离,维护更方便。 对运行环境要求低,基本上一般的

下载
  • 安全写法:在取值后立即判断是否越界,再决定返回还是抛 StopIteration
  • 不要在生成器函数里手动 raise StopIteration —— return 就够了,否则会触发 RuntimeWarning
  • 协程或异步迭代器(__aiter__/__anext__)中,对应抛的是 StopAsyncIteration

如何让自定义类支持 inlen() 和解包而不重复实现

支持 in 查找靠 __contains__,支持 len()__len__,支持解包(如 a, b = obj)靠 __iter__。三者互不替代,也不能靠其中一个自动推导另一个。

典型坑是:只实现了 __iter__ 就以为 in 会自动变快——其实默认回退到逐个 __next__ 对比,时间复杂度 O(n);而自己写 __contains__ 可以用哈希、二分或索引优化到 O(1) 或 O(log n)。

  • __len__ 必须返回非负整数,返回负数会引发 ValueError
  • 解包要求对象可迭代且长度明确(否则报 ValueError: not enough values to unpack
  • 如果底层数据支持快速查找(如 set 或带索引的 dict),别省那几行代码,直接实现 __contains__

用生成器函数代替手写迭代器类真的更“工程”吗

绝大多数情况下是的,但前提是逻辑不依赖外部状态重入或并发访问。生成器函数(含 yield)本质是语法糖,Python 自动帮你管理了 __iter____next__,还内置了状态挂起/恢复机制。

容易被忽略的点是:生成器对象不可重用。调用一次 list(gen()) 后,再次传给 sum(gen()) 就是空的——因为它不是可迭代对象,而是迭代器。所以若需要多次遍历,得包装一层类,或每次调用生成器函数重新创建。

  • 适合场景:一次性数据流(日志行、API 分页结果)、简单变换(map 类逻辑)
  • 不适合场景:需暂停/恢复多个独立遍历(如双指针算法)、需共享内部缓存(如预加载下一页)
  • 调试难点:生成器内部变量无法在暂停时 inspect,不如类属性直观

真正难的不是写对,而是想清楚“这个对象的生命周期归谁管”——是调用方每次要新拿一个,还是它自己能反复交出新迭代器。这点一旦错,下游所有 forlist()itertools 组合都会静默失效。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

816

2023.08.22

golang map内存释放
golang map内存释放

本专题整合了golang map内存相关教程,阅读专题下面的文章了解更多相关内容。

75

2025.09.05

golang map相关教程
golang map相关教程

本专题整合了golang map相关教程,阅读专题下面的文章了解更多详细内容。

36

2025.11.16

golang map原理
golang map原理

本专题整合了golang map相关内容,阅读专题下面的文章了解更多详细内容。

67

2025.11.17

java判断map相关教程
java判断map相关教程

本专题整合了java判断map相关教程,阅读专题下面的文章了解更多详细内容。

46

2025.11.27

页面置换算法
页面置换算法

页面置换算法是操作系统中用来决定在内存中哪些页面应该被换出以便为新的页面提供空间的算法。本专题为大家提供页面置换算法的相关文章,大家可以免费体验。

455

2023.08.14

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

473

2026.02.13

微博网页版主页入口与登录指南_官方网页端快速访问方法
微博网页版主页入口与登录指南_官方网页端快速访问方法

本专题系统整理微博网页版官方入口及网页端登录方式,涵盖首页直达地址、账号登录流程与常见访问问题说明,帮助用户快速找到微博官网主页,实现便捷、安全的网页端登录与内容浏览体验。

158

2026.02.13

Flutter跨平台开发与状态管理实战
Flutter跨平台开发与状态管理实战

本专题围绕Flutter框架展开,系统讲解跨平台UI构建原理与状态管理方案。内容涵盖Widget生命周期、路由管理、Provider与Bloc状态管理模式、网络请求封装及性能优化技巧。通过实战项目演示,帮助开发者构建流畅、可维护的跨平台移动应用。

64

2026.02.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 22.4万人学习

Django 教程
Django 教程

共28课时 | 4.3万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.6万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号