Python 可迭代对象、迭代器、生成器
可迭代对象
定义
在Python的任意对象中,只要它定义了可以返回一个迭代器的 __iter__ 魔法方法,或者定义了可以支持下标索引的 __getitem__ 方法,那么它就是一个可迭代对象,通俗的说就是可以通过 for 循环遍历了。Python 原生的列表,字符串,元祖,字典等都是可以迭代的,可以通过 dir() 函数查看,如以下查看字符串,可见其定义了 __getitem__ 方法

判断一个对象是否是可迭代对象
方法一:isinstance + Iterable
方法二:hasattr + __getitem__
from collections import Iterable
# 方法一
print(isinstance([], Iterable)) # 返回 True,说明是可迭代对象
# 方法二
print(hasattr({}, '__getitem__')) # 返回 True,说明是可迭代对象
自定义可迭代对象
class Employee:def __init__(self, employee):self.employee = employeeemp = Employee(['zs', 'ls', 'ww'])# 正常情况下会报错,emp 不是一个可迭代对象
# TypeError: 'Employee' object is not iterable
for i in emp:print(i)
实现 __iter__ 和 __getitem__ 方法中的一个或者两个
class Employee:def __init__(self, employee):self.employee = employeedef __getitem__(self, item):# item 是解释器帮我们维护的索引值# 在 for 循环中,自动从 0 开始计算print(item)return self.employee[item]emp = Employee(['zs', 'ls', 'ww'])# 正常情况下会报错,emp 不是一个可迭代对象
# TypeError: 'Employee' object is not iterable
for i in emp:print(i)
迭代器
迭代器对象
迭代器就是同时实现了 __next__ 和 __iter__ 方法(缺一不可)的对象。其中 __iter__ 方法返回迭代器自身, __next__ 方法不断返回迭代器中的下一个值,直到容器中没有更多的元素时,抛出 StopIteration 异常,以终止迭代 由定义可以知道,迭代器一定是可迭代对象,因为迭代器实现了 __iter__ 方法,满足了可迭代对象的定义。但是可迭代对象不一定是迭代器,因为不一定实现了 __next__ 方法。
判断一个对象是迭代器
from collections import Iterator
print(isinstance(counter, Iterator)) # True
print(isinstance([], Iterator)) # False
为什么有了可迭代对象,还要有迭代器?
因为迭代器采用了工厂模式,节约了内存空间。所谓的工厂模式,就是在需要的时候,才会去生产数据,而不是像可迭代对象那样一次性 全部把数据生产出来。可迭代对象如列表,字典等,会事先把所有的数据生产并保存起来,而迭代器则是在每次获取下一个值的时候 才会返回值,所以迭代器没有长度这一说法,没有长度这一属性,如果获取完了会抛出 StopIteration 异常来表示没有数据了
所以迭代器适用于那种无限序列,不会占用很大的内存空间
from itertools import count# count 是一个迭代器
# 创建一个从 10 开始的无限序列
counter = count(start=10)
print(type(counter)) # <class 'itertools.count'>
print(dir(counter)) # 有 __iter__ 和 __next__ 方法print(next(counter)) # 10
print(next(counter)) # 11# 报错,没有长度属性
# TypeError: object of type 'itertools.count' has no len()
# print(len(counter))
可迭代对象转为迭代器
a = [1, 2, 3, 4]
print(type(a)) # <class 'list'>a_iter = iter(a)
print(type(a_iter)) # <class 'list_iterator'>
迭代器的特点
a = [1, 2, 3, 4]
a_iter = iter(a)
# 每一次获取迭代器中的值之后,都是把该值从迭代器中拿出来,即迭代器中已经没有该值了,因为已经被拿出来了
# 所以遍历完一次迭代器之后,不能遍历第二次
# 因为遍历第一次的时候已经把值拿出来完了,第二次去遍历的时候迭代器中啥也没有了
# 即迭代器不走回头路,可迭代对象则没有这种特点
for i in a_iter:print(i)for i in a_iter:print(i)# 上面代码只会输出一次 1 2 3 4# 前面说的,当遍历完迭代器的最后一个数据时,再获取数据抛出异常,
# 而 for 循环没有抛出的原因是
# for 循环内部也是用 next() 方法获取迭代器中的值,
# 当抛出异常后 for 循环会对异常进行处理,所以我们采用 for 循环遍历迭代器时不会有异常抛出
生成器
生成器其实是一种特殊的迭代器,但是这种迭代器更加优雅,因为不需要再像普通迭代器那样定义 __next__ 和 __iter__ 方法,只需要一个 yield 关键字,就会自动在内部帮我们实现这两个方法。所以,如果一个函数包含一个或多个 yield 关键字, 这个函数就会变为一个生成器。因为生成器是特殊的迭代器,所以它具备迭代器具备的特性
def demo():print('hello')yield 5print('world')print(type(demo())) # <class 'generator'>
print(dir(demo)) # 有 __iter__ 和 __next__ 方法
yield 关键字的作用
1、程序每次在代码中遇到 yield 关键字后,会返回结果,相当于 return 5,但是并没有真的退出程序,而是保留当前函数的运行状态
2、返回结果后,保留当前函数的运行状态,等待下一次调用,下次调用时,从上一次返回结果处开始执行
第二个作用非常重要,这意味着程序控制权的转移是临时和自愿的,函数将来还会收回控制权(在下次调用生成器的时候收回), 这也是 yield 和 return 最大的区别。return 意味着函数彻底交出控制权并结束运行,下一次调用将固定从函数的第一行代码开始执行。
def demo():print('hello')yield 5print('world')# 此时运行demo(),相当于调用函数,但是并不像普通函数那样马上执行,可以看控制台没有输出
# 此处只是生成一个生成器对象
c = demo()# 利用 next 方法调用生成器
# 第一次调用,打印 hello ,并返回 5
print(next(c))
# 第二次调用,打印 world ,并抛出异常 StopIteration(迭代器特性)
# 因为已经没有语句可以执行了,相当于数据已经取完了
print(next(c))
通过 send 向生成器传递数据
send 方法作用:
1、像 next 方法一样去调用生成器(调用生成器的两种方法:next 方法和 send 方法)
2、send 方法在调用生成器时,可以同时给生成器传递数据到生成器内部
def demo():print('hello')# 注意,此处不是把 yield 5 赋值给变量 t# yield 5 是返回给调用者的值,即返回给 next(c),# 所以执行 print(next(c)) 语句时会在输出 hello 后输出 5# 而变量 t 是接收下一次调用(c.send('test'))时 send 方法传入的 test# 即 t = 'test't = yield 5print('world')print(t)c = demo()
# 第一次调用生成器,得到 t = yield 5 等号左边的表达式结果,即得到 5
print(next(c))
# 第二次调用,从 t = yield 5 语句开始执行,通过 send 方法把 test 传递给 t = yield 5 等号右边的变量 t
# 然后接着执行函数中两个打印语句,输出 world 和 test
# 打印之后就会抛出异常 StopIteration,因为已经没有要执行的语句了
c.send('test')
生成器的预激活机制
def demo():print('hello')t = yield 5print('world')print(t)c = demo()
# next(c) 就是生成器的预激活机制,即第一次调用
# 第一次调用也可以通过 send 方法,但是参数必须为 None
# 因为生成器没有办法在没有激活的情况下接收一个参数
# print(next(c)) # 预激活方式一
c.send(None) # 预激活方式二
c.send('test')
查看生成器的运行过程
将下面代码的每一行打上断点(如图),进入debug模式可以清晰的看到代码的运行过程。

def countdown(n):print('counting down from ', n) # 只在第一次调用生成器的时候输出,此时 n = 10while n >= 0:# 记住,这里不是将 yield n 赋值给变量 newvalue# newvalue 是用来接收 send 方法传递过来的参数的# yield n 是用来将 n 返回给调用者(next 方法或 send 方法)的newvalue = yield n# 判断是否用 send 方法传递参数,如果调用 next 方法,则传递过来的是 Noneif newvalue is not None:n = newvalueelse:n -= 1# 获得生成器对象 c,把 10 传递给 n
c = countdown(10)for i in c:# 第一次调用生成器# 第一次执行 for i in c 语句时,内部第一次调用 next(c)# 执行 print('counting down from ', n) 语句,此时 n=10# 然后进入 while n >= 0 循环,执行 newvalue = yield n 等号右边的 yield n,即返回 n=10 给i# 所以第一次for循环 i=10,输出 10# 第三次调用生成器# 第二次执行 for i in c 语句时,内部第二次调用 next(c),此时已经第三次调用生成器,第二次是用 send 方法调用的# 接着第二次调用生成器的地方开始执行,即从 newvalue = yield n 开始执行,next 方法没有给生成器传递参数# 所以 newvalue 为 None,则执行 else 语句:n -= 1,所以 n = 2# 进入 while n >= 0 ,然后返回newvalue = yield n 等号右边的 yield n,即返回 n=2# 所以第二次for循环 i=2,输出 2# 第四次调用生成器# 第三次执行 for i in c 语句时,内部第三次调用 next(c)# 接着第三次调用生成器的地方开始执行,即从 newvalue = yield n 开始执行,next 方法没有给生成器传递参数# 所以 newvalue 为 None,则执行 else 语句:n -= 1,所以 n = 1# 进入 while n >= 0 ,然后返回newvalue = yield n 等号右边的 yield n,即返回 n=1# 所以第三次for循环 i=1,输出 1# 第五次调用生成器# 第四次执行 for i in c 语句时,内部第四次调用 next(c)# 接着第四次调用生成器的地方开始执行,即从 newvalue = yield n 开始执行,next 方法没有给生成器传递参数# 所以 newvalue 为 None,则执行 else 语句:n -= 1,所以 n = 0# 进入 while n >= 0 ,然后返回newvalue = yield n 等号右边的 yield n,即返回 n=0# 所以第四次for循环 i=0,输出 0# 第六次调用生成器,最后一次# 第五次执行 for i in c 语句时,内部第五次调用 next(c)# 接着第五次调用生成器的地方开始执行,即从 newvalue = yield n 开始执行,next 方法没有给生成器传递参数# 所以 newvalue 为 None,则执行 else 语句:n -= 1,所以 n = -1# 不能进入 while n >= 0 ,那么生成器运行到最后结束,抛出一个异常 StopIteration# 抛出的异常被for循环内部处理,所以运行到此结束,for循环也结束print(i) # 分别输出 10 2 1 0# 第一次进入 for 循环时 i = 10,所以进入 if 判断if i == 10:# 第二次调用生成器# 调用 send 方法调用生成器,接着第一次调用的地方继续执行# 把参数传递给 newvalue = yield n 中的 newvalue,此时 newvalue=3# newvalue 不为 None,进入判断 if newvalue is not None,将 n 设置为 newvalue (n = newvalue)# 此时 n = 3,进入 while n >= 0 ,然后返回newvalue = yield n 等号右边的 yield n,即返回 n=3# 然后进行第二次 for 循环print('send: ', c.send(3)) # 打印 send: 3
获得生成器的第二种方式
除了通过 yield 关键字得到生成器外,还可以通过小括号的形式得到,如下
"""
生成器表达式和列表推导式的区别:列表推导式会一下子将所有的数据生产出来,并放到列表中生成器表达式一次只生产一个数据获得生成器的两种方式:1、将普通函数里的 return 替换成 yield ,这样调用函数时会得到一个生成器2、利用生成器表达式
"""# 可以通过元祖表达式(小括号)得到生成器
a = (i for i in range(5))
# 列表推导式得到的是一个列表
b = [i for i in range(5)]# a 是生成器
print(type(a)) # <class 'generator'>
# b 是列表
print(type(b)) # <class 'list'># 利用推导式获得生成器的方式称为生成器表达式
t = (i * 2 for i in range(5))
print(t) # <generator object <genexpr> at 0x000001F466254620>
print(next(t)) # 0
print(next(t)) # 2
# 把剩下的数据处理,目前只剩下 2 3 4
for i in t:print(i) # 4 6 8
生成器实现斐波那契数列
# 用生成器实现斐波那契数列
def fib():num1, num2 = 0, 1while True:yield num1num1, num2 = num2, num1 + num2f = fib()
# 因为 fib 函数里是死循环,所以只要调用 next(f) 就可以一直得到 斐波那契数列的值
print(next(f)) # 0
print(next(f)) # 1
print(next(f)) # 1
print(next(f)) # 2
print(next(f)) # 3
print(next(f)) # 5相关文章:
Python 可迭代对象、迭代器、生成器
可迭代对象 定义 在Python的任意对象中,只要它定义了可以返回一个迭代器的 __iter__ 魔法方法,或者定义了可以支持下标索引的 __getitem__ 方法,那么它就是一个可迭代对象,通俗的说就是可以通过 for 循环遍历了。Python 原生的列…...
HTML的有序列表、无序列表、自定义列表
目录 背景: 过程: 有序列表: 简介: 代码展示: 效果展示: 无序列表: 简介: 代码展示: 效果展示: 自定义列表: 简介: 代码展示: 效果展示: 总结: 背景: 1.有序列表(Ordered List): 有序列表是最早的…...
银河麒麟安装Docker-国产化-九五小庞
银河麒麟高级服务器操作系统 V10 是针对企业级关键业务,适应虚拟化、 云计算、大数据、工业互联网时代对主机系统可靠性、安全性、性能、扩展性和 实时性的需求,依据 CMMI 5 级标准研制的提供内生安全、云原生支持、国产 平台深入优化、高性能、易管理的…...
数据库与身份认证
1. 数据库的基本概念 1.1 什么是数据库 数据库(database)是用来组织、存储和管理数据的仓库。 当今世界是一个充满着数据的互联网世界,充斥着大量的数据。数据的来源有很多,比如出行记录、消费记录、浏览的网页、发送的消息…...
LabVIEW开发锅炉汽包水位的监督控制和模拟
LabVIEW开发锅炉汽包水位的监督控制和模拟 控制锅炉汽包液位对于机械的安全和设备的保护至关重要。滚筒液位控制器的工作是将滚筒液位提高到指定的设定点,并保持在那里,同时保持一致的蒸汽负荷。锅炉管可能会因该水平急剧下降而暴露,这会导致…...
2023-简单点-树莓派安装ncnn框架
not python 按照下面的步骤进行就可以了: 参考 tips: 其中有一步要用下面方法: 如果你的git clone不得行,可以按照以下操作方法: git clone --depth1 https://ghproxy.com/ https://github.com/Tencent/ncnn.git python 直接 pip install …...
Docker核心原理与实操
第一章、Docker基本概念 1、概念:Docker是一种容器技术,可以解决软件跨环境迁移问题。 2、实现原理:是一个分层复用的文件系统;每一层都是一个独立的软件; …...
虚幻引擎 UE5 增强输入系统
用人话讲!虚幻引擎 UE5 增强输入系统(蓝图篇)_酥妃大魔王i的博客-CSDN博客 UE5 -- EnhancedInput(增强输入系统) - 知乎 (zhihu.com) 简单认识 虚幻引擎中的增强输入 | 虚幻引擎5.1文档 (unrealengine.com) 文档有较详细介绍 标记一下方便…...
Mac 安装软件各种报错解决方案
Mac 安装软件各种报错解决方案 文章目录 Mac 安装软件各种报错解决方案一. 打开允许“允许任何来源”二. 无法打开"xxx",因为它不是从App Store下载三. 无法打开"xxx",因为 Apple无法检查其是否包含恶意软件。四. "xxx"已…...
leetcode做题笔记142. 环形链表 II
给定一个链表的头节点 head ,返回链表开始入环的第一个节点。 如果链表无环,则返回 null。 如果链表中有某个节点,可以通过连续跟踪 next 指针再次到达,则链表中存在环。 为了表示给定链表中的环,评测系统内部使用整…...
DuDuTalk:4G语音工牌,如何实现家庭上门维修服务过程的智能化管理?
随着上门按摩、上门养老、上门买菜、上门维修等互联网上门服务的兴起,越来越多的居民开始采用线上下单,享受企业安排人员上门到家的服务。而家庭维修作为到家服务里面典型的一个场景,已成为许多人不可或缺的一部分。然而,与此同时…...
Mybatis常见面试题总结
梳理面试过程中Mybatis相关的常见问题。为保证知识点覆盖,参考了《Mybatis从入门到精通》、《深入浅出Mybatis技术原理与实战》、《Mybatis技术内幕》等书籍。 Mybatis 简介 Mybatis 是一款优秀的持久层框架(ORM框架),它支持自定义SQL、存储过程以及高…...
数字IC设计之时序分析基础概念汇总
1 时钟Clock 理想的时钟模型是一个占空比为50%且周期固定的方波。时钟是FPGA中同步电路逻辑运行的一个基准。理想的时钟信号如下图: 2 时钟抖动Clock Jitter 理想的时钟信号是完美的方波,但是实际的方波是存在一些时钟抖动的。那么什么是时钟抖动呢?时钟抖动&#…...
1.centos7安装docker
本文目录: 1.docker 安装1.安装步骤2.安装是否成功校验3.docker加速配置4.hello world来袭,验证安装是否ok 2.卸载docker3.卸载较旧版本docker 使用docker必备的三个官方网站: docker官网:https://www.docker.com docker官方仓库&…...
基于elasticsearch-8.8.2 kibana-8.8.2 搭建一个文搜图系统demo
数据来源是由 图片url,图片descript,图片keywords 外加一个id 基于此首先创建 索引, keywords是一组由单词或词组 组成的一组数据,所以以数组形式压入数据: descript 是由两条语句组合成的数据(针对图片的两种不同描述) # 这里创建的keywords 数组元素类型为text,即可以模糊匹…...
第26节-PhotoShop基础课程-形状工具组-画板
文章目录 前言1.基础图形1.正方形(shift/alt/两者加起来)2.描边3.合并形状 将多个图放在一个图层4.对齐只能在不同图层5.修改6.重叠,相交 2.多边形1.边数2.星形 3.直线工具1.正常2.箭头 4.自定义形状 前言 类似画图板,矢量图形。…...
第一次课,通过进程信息和服务信息识别当前计算机运行程序(预习版)
题目: 检测的目标进程: ydebugg ; “ImmunityDebugger.exe” _500], rax Exe ; “ollydbg.exe” _4F8], rax hackerE ; “ProcessHacker.exe” _4F0], rax Exe ; “tcpview.exe” _4E8], rax sExe ; “autoruns.exe” _4E0], rax scExe ; “autorunsc.ex…...
ChatGPT 或其它 AI,能用在文书创作上吗?
新的申请季已经正式开始,一些热门项目的ED截止日期也不再遥远,因此很多准留学生们都已经开始了关于文书的创作。 而随着科技的不断发展,以ChatGPT为首的一众AI工具也作为一种辅助手段愈发融入了我们的生活。 那么不免就会有一些同学在准备申…...
Java中锁的分类
Java中锁的分类 这篇文章介绍各种锁的分类。介绍的内容如下: 公平锁/非公平锁可重入锁独享锁/共享锁互斥锁/读写锁乐观锁/悲观锁分段锁偏向锁/轻量级锁/重量级锁自旋锁 上面是很多锁的名词,这些分类并不是全是指锁的状态,有的指锁的特性&am…...
centos安装flink,通过windows访问webui
1. 安装flink 1.1. flink的下载 通过flink官网下载flink安装包 https://flink.apache.org/ 下载安装包 1.2 flink在centos上的安装 将下载好的flink-1.17.1-bin-scala_2.12.tgz安装包放到centos目录下 解压文件: [rootlocalhost ~]# tar -zxvf flink-1.17.…...
7.4.分块查找
一.分块查找的算法思想: 1.实例: 以上述图片的顺序表为例, 该顺序表的数据元素从整体来看是乱序的,但如果把这些数据元素分成一块一块的小区间, 第一个区间[0,1]索引上的数据元素都是小于等于10的, 第二…...
Python:操作 Excel 折叠
💖亲爱的技术爱好者们,热烈欢迎来到 Kant2048 的博客!我是 Thomas Kant,很开心能在CSDN上与你们相遇~💖 本博客的精华专栏: 【自动化测试】 【测试经验】 【人工智能】 【Python】 Python 操作 Excel 系列 读取单元格数据按行写入设置行高和列宽自动调整行高和列宽水平…...
《Playwright:微软的自动化测试工具详解》
Playwright 简介:声明内容来自网络,将内容拼接整理出来的文档 Playwright 是微软开发的自动化测试工具,支持 Chrome、Firefox、Safari 等主流浏览器,提供多语言 API(Python、JavaScript、Java、.NET)。它的特点包括&a…...
蓝桥杯 2024 15届国赛 A组 儿童节快乐
P10576 [蓝桥杯 2024 国 A] 儿童节快乐 题目描述 五彩斑斓的气球在蓝天下悠然飘荡,轻快的音乐在耳边持续回荡,小朋友们手牵着手一同畅快欢笑。在这样一片安乐祥和的氛围下,六一来了。 今天是六一儿童节,小蓝老师为了让大家在节…...
学校招生小程序源码介绍
基于ThinkPHPFastAdminUniApp开发的学校招生小程序源码,专为学校招生场景量身打造,功能实用且操作便捷。 从技术架构来看,ThinkPHP提供稳定可靠的后台服务,FastAdmin加速开发流程,UniApp则保障小程序在多端有良好的兼…...
Neo4j 集群管理:原理、技术与最佳实践深度解析
Neo4j 的集群技术是其企业级高可用性、可扩展性和容错能力的核心。通过深入分析官方文档,本文将系统阐述其集群管理的核心原理、关键技术、实用技巧和行业最佳实践。 Neo4j 的 Causal Clustering 架构提供了一个强大而灵活的基石,用于构建高可用、可扩展且一致的图数据库服务…...
UR 协作机器人「三剑客」:精密轻量担当(UR7e)、全能协作主力(UR12e)、重型任务专家(UR15)
UR协作机器人正以其卓越性能在现代制造业自动化中扮演重要角色。UR7e、UR12e和UR15通过创新技术和精准设计满足了不同行业的多样化需求。其中,UR15以其速度、精度及人工智能准备能力成为自动化领域的重要突破。UR7e和UR12e则在负载规格和市场定位上不断优化…...
【学习笔记】深入理解Java虚拟机学习笔记——第4章 虚拟机性能监控,故障处理工具
第2章 虚拟机性能监控,故障处理工具 4.1 概述 略 4.2 基础故障处理工具 4.2.1 jps:虚拟机进程状况工具 命令:jps [options] [hostid] 功能:本地虚拟机进程显示进程ID(与ps相同),可同时显示主类&#x…...
【HarmonyOS 5 开发速记】如何获取用户信息(头像/昵称/手机号)
1.获取 authorizationCode: 2.利用 authorizationCode 获取 accessToken:文档中心 3.获取手机:文档中心 4.获取昵称头像:文档中心 首先创建 request 若要获取手机号,scope必填 phone,permissions 必填 …...
RNN避坑指南:从数学推导到LSTM/GRU工业级部署实战流程
本文较长,建议点赞收藏,以免遗失。更多AI大模型应用开发学习视频及资料,尽在聚客AI学院。 本文全面剖析RNN核心原理,深入讲解梯度消失/爆炸问题,并通过LSTM/GRU结构实现解决方案,提供时间序列预测和文本生成…...
