Python 中的生成器实现原理是什么?
off999 2024-12-12 14:22 17 浏览 0 评论
本文首发自「慕课网」,想了解更多IT干货内容,程序员圈内热闻,欢迎关注!
作者| 慕课网精英讲师 朱广蔚
1. 如何生成一个巨大的序列
1.1 需求描述
要求生成一个包含很多元素的序列,假设:
- 存储 1 个整数需要 4 个字节
- 现在要创建一个包含 1 G 个整数的序列,从 0 到 1 * 1024 * 1024 * 1024 - 1
- 如果需要为序列中的每个整数分配内存,则需要分配的内存为 1G * 4 = 4G
1.2 通过列表推导
Python 提供了列表推导用于生成列表,下面使用列表推导生成一个包含 0 到 4 之间所有整数的列表,代码如下:
>>> list = [i for i in range(4)]
>>> list
[0, 1, 2, 3]
代码块123- 在第 1 行,使用列表推导创建一个包含 4 个元素的列表
- 在第 2 行,显示新创建的列表
- 在第 3 行,创建了一个包含 0、1、2、3 等 4 个元素的列表
如果生成一个从 0 到 1G 的列表,代码如下:
>>> N = 1024 * 1024 * 1024
>>> list = [i for i in range(N)]
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<stdin>", line 1, in <listcomp>
MemoryError
代码块123456- 在第 1 行,设定 N 为 1G
- 在第 2 行,使用列表推导创建一个包含 N 个元素的列表
- 在第 6 行,程序运行出错,提示 MemoryError
使用列表推导创建包含 1G 个整数的列表时,需要为这 1G 个整数分配至少 4G 的内存,需要消耗大量的内存,超出了 Python 的限制,因此出现了 MemoryError 的错误。
另外,创建这个巨大的列表需要消耗大量的时间,因此执行第 2 行的语句后,系统失去响应,大约 10 多秒后才出现错误信息。
1.3 通过动态计算
列表推导需要一次性的为 1G 个整数分配内存空间,带来了两个问题:
- 列表占用了大量的物理内存
- 创建列表的时间过长
Python 提供了一种动态计算的思路解决以上问题,它的思想如下:
- 要生成的序列是有规则的,在这个例子中,要求生成连续递增的序列
- 使用一个特殊的对象 generator,该对象被称为生成器 generator,生成器按照规则依次输出该序列
- Python 提供了内置方法 next(generator),该方法通知生成器产生下一个数据并返回该数据
- 不需要为 generator 预先分配内存,通过调用 next(generator) 可以动态获取序列的下一个数据
创建一个输出从 0 到 1G 的生成器,代码如下:
>>> N = 1024 * 1024 * 1024
>>> generator = (i for i in range(N))
>>> next(generator)
0
>>> next(generator)
1
>>> next(generator)
2
代码块12345678- 在第 1 行,设定 N 为 1G
- 在第 2 行,使用类似于列表推导的语法创建一个生成器,它输出从 0 到 1G 的序列注意:创建生成器的语法采用小括号 (),创建列表的语法采用方括号 []
- 在第 3 行,使用 next(generator),通知 generator 生产一个数据在第 4 行,generator 输出从 0 到 1G 序列中的第 0 个整数
- 在第 5 行,使用 next(generator),通知 generator 生产一个数据在第 6 行,generator 输出从 0 到 1G 序列中的第 1 个整数
- 在第 7 行,使用 next(generator),通知 generator 生产一个数据在第 8 行,generator 输出从 0 到 1G 序列中的第 2 个整数
注意:在第 2 行,创建一个输出从 0 到 1G 的序列的生成器,因为不需要分配内存,创建生成器的速度非常快,几乎是瞬间完成的。与之相比,在上一节中创建一个输出从 0 到 1G 的序列的列表,因为需要分配内存,创建列表的速度非常慢,并且导致了 MemoryError。
2. 生成器概述
2.1 生成器的定义
在 Python 中,生成器是一个特殊的对象,它按照一定的规则依次输出数据。Python 的内置函数 next(generator) 通知生成器输出一个新的数据,当生成器输出全部数据后,产生一个特殊的异常 StopIteration,用于标记生成器输出结束。
下面的代码创建一个产生 0 到 3 之间所有整数的生成器:
>>> generator = (i for i in range(3))
>>> next(generator)
0
>>> next(generator)
1
>>> next(generator)
2
>>> next(generator)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
StopIteration
代码块1234567891011- 在第 1 行,创建一个产生 0 到 3 之间所有整数的生成器注意:创建生成器的语法采用小括号 (),创建列表的语法采用方括号 []
- 在第 2 行,生成器产生第 0 个整数
- 在第 4 行,生成器产生第 1 个整数
- 在第 6 行,生成器产生第 2 个整数
- 在第 8 行,生成器产生第 3 个整数在第 11 行,因为生成器生成的序列只包含 3 个整数,此时已经生成全部的整数,因此抛出异常 StopIteration
2.2 使用 while 循环访问生成器
根据生成器的原理,可以循环的调用 next(generator) 输出全部的序列,示例如下:
generator = (i for i in range(3))
while True:
try:
item = next(generator)
print(item)
except StopIteration:
break
代码块12345678- 在第 1 行,创建一个产生 0 到 3 之间所有整数的生成器
- 在第 3 行,创建一个循环在第 5 行,调用 next(generator) 通知生成器返回一个数据在第 7 行,当生成器输出结束后,抛出异常 StopIteration
运行程序,输出结果如下:
0
1
2
代码块1232.3 使用 for 循环访问生成器
通常使用 for 循环访问生成器,示例如下:
generator = (i for i in range(3))
for item in generator:
print(item)
代码块1234- 在第 1 行,创建一个产生 0 到 3 之间所有整数的生成器
- 在第 3 行,使用 for 循环访问生成器
运行程序,输出结果如下:
0
1
2
代码块1233. 创建生成器
3.1 通过推导创建生成器
可以使用类似于列表推导的语法创建一个生成器,语法如下:
(expression for i in iterable)
代码块1该生成器遍历对象 iterable,依次产生数据 expression,它的工作流程如下:
for i in iterable:
generate expression
代码块12注意:创建生成器的语法与列表推导的语法相似,不同之处在于,创建生成器的语法采用小括号 (),创建列表的语法采用方括号 []。
通过推导创建生成器的示例如下:
generator = (i*2 for i in range(5))
for i in generator:
print(i)
代码块123- 循环变量 i 从 0 变化到 4
- 生成器每次产生数据 i*2
运行程序,输出结果如下:
0
2
4
6
8
代码块123453.2 通过复杂的推导创建生成器
可以使用类似于列表推导的语法创建一个生成器,语法如下:
(expression for i in iterable if condition)
代码块1该生成器遍历对象 iterable,如果条件 condition 为真,则产生数据 expression,它的工作流程如下:
for i in iterable:
if condition:
generate expression
代码块123通过复杂推导创建生成器的示例如下:
generator = (i for i in range(10) if i % 2 == 0)
for i in generator:
print(i)
代码块123- 循环变量 i 从 0 变化到 9
- 如果 i % 2 == 0,表示 i 是偶数
- 生成器每次产生从 0 到 9 之间的偶数
运行程序,输出结果如下:
0
2
4
6
8
代码块123453.3 通过 yield 创建生成器
在生成器的生命周期中,生成器根据一定的规则产生一系列的数据,生成器可以使用 yield 关键字产生一个数据。例如,一个生成特定范围内的奇数序列的函数:
def generateOddNumbers(n):
for i in range(n):
if i % 2 == 1:
yield i
generator = generateOddNumbers(10)
for i in generator:
print(i)
代码块12345678- 在第 1 行,定义了函数 generateOddNumbers(n),它返回一个生成器,该生成器产生从 0 到 n 范围内的奇数
- 在第 2 行到第 4 行,使用 for 循环生成从 0 到 n 范围内的奇数在第 3 行,如果 i % 2 == 1 为真,表示 i 是奇数在第 4 行,使用 yield i 生成一个数据 i
- 在第 6 行,generateOddNumbers(10) 返回一个生成器,该生成器产生从 0 到 10 范围内的奇数
- 在第 7 行,使用 for 循环遍历该生成器
运行该程序,输出如下:
1
3
5
7
9
代码块12345注意:包含 yield 关键字的函数被称为生成器函数,调用生成器函数会返回一个生成器。在上面的例子中,函数 generateOddNumbers(n) 包含 yield 关键字,是一个生成器函数,它返回一个生成器,该生成器产生从 0 到 n 范围内的奇数。
4. 使用 yield 实现遍历堆栈的生成器
4.1 通过单链表实现堆栈
通过单链表实现堆栈,图示如下:

在上图中,每个节点有两个字段: item 和 next,item 用于存储数据,next 指向下一个节点,head 指针指向堆栈的顶部。描述堆栈的 Python 代码如下:
class Node:
def __init__(self, item):
self.item = item
self.next = None
class Stack:
def __init__(self):
self.head = None
def push(self, item):
node = Node(item)
node.next = self.head
self.head = node
stack = Stack()
stack.push('a')
stack.push('b')
stack.push('c')
代码块123456789101112131415161718- 在第 1 行,定义了类 Node 用于描述链表中的节点
- 在第 6 行,定义了类 Stack 描述堆栈在第 8 行,定义了头指针 head,指向链表中的首个节点在第 10 行,定义了成员方法 push,将元素压如到堆栈中在第 11 行,创建一个新节点 node在第 12 行,新节点 node 的 next 指向头结点在第 13 行,头结点指向新节点
- 在第 15 行,创建一个对象 stack
- 在第 16 行到第 18 行,依次压入 3 个元素 ‘a’、‘b’、‘c’
4.2 使用 yield 关键字实现生成器函数
def stackGenerate(stack):
cursor = stack.head
while cursor != None:
yield cursor.item
cursor = cursor.next
代码块12345- 在第 1 行,定义函数 stackGenerate(stack)该函数包含 yield 关键字,是一个生成器函数,它返回一个生成器生成器遍历堆栈,按出栈的顺序输出数据
- 在第 2 行,变量 cursor 指向了当前正在遍历的元素,初始化被设置为链表的头结点
- 在第 3 行,使用循环遍历堆栈如果变量 cursor 等于 None,表示已经到达链表的尾部,即遍历完全部的元素了在第 4 行,使用 yield 输出当前正在遍历的元素在第 5 行,将 cursor 指向下一个元素
4.3 通过 while 循环遍历堆栈
使用 while 循环显式的使用 next、StopIteration 完成对 stack 的遍历,代码如下:
generator = stackGenerate(stack)
while True:
try:
item = next(generator)
print(item)
except StopIteration:
break
代码块1234567- 在第 1 行,stackGenerate(stack) 返回一个遍历堆栈的生成器
- 在第 4 行,next(generator) 获取生成器的输出
- 在第 6 行,当生成器输出结束后,抛出异常 StopIteration
程序依次压入 ‘a’、‘b’、‘c’,遍历时以压入相反的顺序输出,结果如下:
c
b
a
代码块1234.4 通过 for … in 循环遍历堆栈
通过 for … in 循环对生成器进行遍历,代码如下:
generator = stackGenerate(stack)
for item in generator:
print(item)
代码块123与上一节的代码相比,代码要简洁很多,程序输出相同的结果如下:
c
b
a
代码块123欢迎关注「慕课网」,发现更多IT圈优质内容,分享干货知识,帮助你成为更好的程序员!
相关推荐
- win11家庭版和专业版哪个好(win11家庭版和专业版哪个好用)
-
win11用专业版本最好,玩游戏最稳定。相对于前代的操作系统来讲,windowS11系统无论哪个版本都非常稳定,但是在门斗41所有版本当中,最稳定的是专业版,但是这个版本占用的空间很大,对硬件的要求...
- win7连不上win10打印机拒绝访问
-
原因及解决方法:1.在Win10Cortana搜索框中,搜索控制面板将打开;2.转到控制面板,选择查看方法作为一个小图标,然后单击程序和功能;3.单击以启用或禁用Windows功能选项;4....
- 电脑主机不启动但通电(电脑主机开机不通电没反应)
-
第1步:首先检查电脑的外部接线是否接好,把各个连线重新插一遍,看故障是否排除。第2步:如果故障依旧,接着打开主机箱查看机箱内有无多余金属物,或主板变形造成的短路,闻一下机箱内有无烧焦的糊味,主板上有无...
- 苹果一体机怎么重装系统(苹果一体机重装系统win10)
-
苹果更换硬盘后,如果您没有创建一个启动磁盘或者使用TimeMachine备份系统,您可以按照以下步骤重新安装操作系统:1.准备一个可用于安装系统的启动磁盘。您可以使用官方提供的macOS安装介质(...
-
- 装系统的u盘怎么制作(装系统怎么制作u盘启动盘)
-
1、下载安装u当家u盘启动盘制作工具,完成之后打开该软件2、将之前准备好的U盘插入电脑,U当家会自动识别并选为默认,如果有多个U盘可以在“选择U盘”的下拉框中选择要制作成U盘启动盘的U盘。3、点击下面的“一键制作”按钮。(如果是电脑主板是U...
-
2025-11-15 07:03 off999
- 台电u盘怎么样
-
感觉台电U盘还是可以的,我知道台电是国内最成功的数码产品制造商之一,是一个致力于IT及消费数码类产品研发、生产、销售及服务的一体化品牌,U盘很好用,用的人很多。威刚是480Mbit/S只是USB2...
- u盘重装系统蓝屏进不去(u盘装系统重启蓝屏)
-
尊敬的用户您好:可按下面的步骤进行系统盘装系统:1.买一张win7光盘,将系统光盘放入光驱里。2.打开电源,然后观察屏幕的提示,在启动系统之前有按“f2进行bios设置。3.设置启动方式。找到boot...
- win快捷键大全(win快捷键大全表)
-
win快捷键:单独按Windows:显示或隐藏“开始”功能表Windows+BREAK:显示“系统属性”对话框Windows+D:显示桌面或恢复桌面Windows+M:最小化所有窗口Windows...
- u盘突然要格式化怎么回事(用着的u盘突然要格式化)
-
U盘提示格式化的原因可能有以下几点:文件系统损坏:U盘的文件系统可能出现损坏,导致无法正常读取和写入数据。病毒感染:U盘可能被病毒感染,病毒会破坏文件系统或隐藏文件。不兼容的文件系统:U盘可能使用了一...
- 路由器pppoe怎么设置(路由器如何设置pppoe上网)
-
移动宽带PPPOE是不用设置的,你只要把帐号和密码填上其他都不选就能上网。但是要注意连接到你电脑的网线是直接经过一个路由器后连接到猫上的,因为我见过有人是这样连接的猫---路由器---路由器---电脑...
- win7桌面图标设置在哪里(win7桌面图标在哪个文件夹)
-
要调出Win7桌面图标,首先需要进入桌面。可以通过点击任务栏上方的“显示桌面”按钮或按下Windows键+D快捷键进入桌面。在桌面上右键单击空白区域,选择“个性化”,然后选择“更改桌面图标”选项。在弹...
- win7手机壁纸(windows7手机桌面主题)
-
1.首先我们打开手机进入到桌面,找到设置图标点击打开。华为手机如何设置自定义锁屏壁纸2.进入到设置界面之后,点击下方的显示选项。华为手机如何设置自定义锁屏壁纸3.然后我们就可以在显示界面中找到壁纸...
- 怎么开启路由器wifi(怎么开启路由器的dhcp功能)
-
把路由器改为开放网络方法如下:1、打开浏览器,在地址栏输入路由器网关IP地址(路由器背后的标签上有,一般是192.168.1.1),输入登录用户名和密码(一般均为admin);2、登录成功后就显示运行...
- win10加载不出来桌面黑屏(window10加载不出来)
-
方法一、如果是遇到了突发性黑屏首先尝试使用Ctrl+Alt+Delete组合键来进行电脑重启一般重启可以解决大部分黑屏故障。win10电脑黑屏按什么键恢复_win10电脑黑屏一键恢复的方法方法...
- 怎么用火绒关闭win安全中心(win10火绒安全软件怎么关闭)
-
一、火绒防火墙关闭方法在电脑上运行火绒软件后,点击首页界面的“防护中心”,在病毒防护界面可以关闭文件、U盘、邮件等保护功能,。系统防护:在该界面可以关闭系统防护功能,。网络防护:可以关闭网络保护等功...
欢迎 你 发表评论:
- 一周热门
-
-
抖音上好看的小姐姐,Python给你都下载了
-
全网最简单易懂!495页Python漫画教程,高清PDF版免费下载
-
Python 3.14 的 UUIDv6/v7/v8 上新,别再用 uuid4 () 啦!
-
python入门到脱坑 输入与输出—str()函数
-
飞牛NAS部署TVGate Docker项目,实现内网一键转发、代理、jx
-
宝塔面板如何添加免费waf防火墙?(宝塔面板开启https)
-
Python三目运算基础与进阶_python三目运算符判断三个变量
-
(新版)Python 分布式爬虫与 JS 逆向进阶实战吾爱分享
-
慕ke 前端工程师2024「完整」
-
失业程序员复习python笔记——条件与循环
-
- 最近发表
- 标签列表
-
- python计时 (73)
- python安装路径 (56)
- python类型转换 (93)
- python进度条 (67)
- python吧 (67)
- python的for循环 (65)
- python格式化字符串 (61)
- python静态方法 (57)
- python列表切片 (59)
- python面向对象编程 (60)
- python 代码加密 (65)
- python串口编程 (77)
- python封装 (57)
- python写入txt (66)
- python读取文件夹下所有文件 (59)
- python操作mysql数据库 (66)
- python获取列表的长度 (64)
- python接口 (63)
- python调用函数 (57)
- python多态 (60)
- python匿名函数 (59)
- python打印九九乘法表 (65)
- python赋值 (62)
- python异常 (69)
- python元祖 (57)
