Python零基础从小白打怪升级中~~~~~~~生成器和迭代器

devtools/2024/12/22 16:13:35/

第十七节:生成器和迭代器

一、迭代器

本质: 一个实现了__iter__方法和__next__方法的对象

注意 Iterator对象和 Iterable对象,一个是迭代器,一个是可迭代对象

1、list、dict、str、tuple、set是可迭代对象但不是迭代器;

2、可迭代对象可以转为迭代器,for循环会自动转换成迭代器。或者调用iter函数

3、如果把所有数据丢到列表中 可以 优点 速度快 缺点 列表占内存太大,如果使用迭代器申请固定的空间也就是一个个的拿出来, 能节约内存,但是浪费时间;

4、需要用类来写迭代器,需要重写 _ iter _( ) 和 _ next _( )方法;思考一下

5、自定义迭代器最大的特点是,需要用类来写,显得代码冗长,不太方便。所以直接使用生成器

python">from collections.abc import Iterable, Iterator
class GenreratorPrime(object):def __init__(self):self.i = 2# 需要用类来写迭代器,需要重写 _ _iter_ _( ) 和 _ _next_ _( )方法;def __iter__(self):return selfdef __next__(self):if self.i == 2:self.i += 1return 2while True:self.i += 1for j in range(2, self.i):if self.i % j == 0:breakelse:return self.igp = GenreratorPrime()
print(isinstance(gp, Iterable))
print(isinstance(gp, Iterator))
iter1 = iter(gp)
print(next(iter1))
print(next(iter1))
print(next(iter1))
print(next(iter1))
print(next(iter1))
print(next(iter1))
print(next(iter1))
print(next(iter1))

二、生成器

生成器(generator)也是一种迭代器 ,在每次迭代时返回一个值,直到抛出 StopIteration 异常。它有两种构造方式:

  • 表达式来创建生成器
  • 包含有yield的函数来创建生成器

1、表达式创建生成器

python">print([x for x in range(6)])
print((x for x in range(6)))
numbers = (x for x in range(6))
# for n in numbers:
#     print(n)print(hasattr(numbers, '__iter__'))
print(hasattr(numbers, 'next'))
print(hasattr(numbers, '__next__'))print(numbers.__next__())print(next(numbers))
print(next(numbers))
print(next(numbers))
print(next(numbers))
print(next(numbers))
print(next(numbers))
print(next(numbers))

总结:

可以看出生成器表达式无法像列表推导式那样直接输出,它和可迭代对象一样只能采用for循环调用next()函数,原因在于range返回的是一个可迭代对象,列表推导式之所以能直接print就是因为[]将可迭代对象转为列表。

2、含有yield关键字的函数

一个带有 yield 的函数就是一个生成器函数,当我们使用 yield 时,它帮我们自动创建了__iter__() 和 next() 方法,而且在没有数据时,也会抛出 StopIteration 异常,也就是我们不费吹灰之力就获得了一个迭代器,非常简洁和高效。

python">def generator_func():v1 = yield 1print(f'hello {v1}')v1 = yield 2print(f'value1 is {v1}')v2 = yield 3print(f'value2 is {v2}')v3 = yield 4print(f'value3 is {v3}')g = generator_func()
print(g.__next__())
print(g.__next__())
print(g.send(100))
g.send(1)
print(g.send(2))

总结:

  1. yield 把函数变成了一个生成器。
  2. 调用该函数的时候不会立即执行代码,而是返回了一个生成器对象
  3. 当使用 next() (在 for 循环中会自动调用 next() ) 作用于返回的生成器对象时,函数 开始执行,在遇到 yield 的时候会『暂停』,并返回当前的迭代值;
  4. 当再次使用 next() 的时候,函数会从原来『暂停』的地方继续执行,直到遇到 yield语 句,如果没有 yield 语句,则抛出异常;
  5. 生成器函数的执行过程看起来就是不断地 =执行->中断->执行->中断的过程
  6. send() 方法就是 next() 的功能,加上传值给 上次暂停的yield 。
  7. close() 方法来关闭一个生成器。生成器被关闭后,再次调用 next() 方法,不管能否遇到 yield 关键字,都会抛出 StopIteration 异常,

3、案例

创建一个获取所有质数的生成器

python">def generator_prime():i = 2yield iwhile True:i += 1for j in range(2, i):if i % j == 0:breakelse:yield ig = generator_prime()
print(next(g))
print(next(g))
print(next(g))
print(next(g))
print(next(g))
print(next(g))
print(next(g))

http://www.ppmy.cn/devtools/2215.html

相关文章

Kafka 知识汇总学习

kafka:消息发布队列、具有存储的功能 生产者 消费者 优势:吞吐量高,性能好; 具有良好的伸缩性,支持在线水平扩展; 具有容错性和可靠性; 与大数据生态结合 Kafka 是一个分布式系统,由服务器和客户端组成&…

新型物联网创新实践教学体系建设

新型物联网创新实践教学体系建设 一、设计背景 随着物联网技术的快速发展,物联网已成为当今科技创新的重要领域。为了培养能够紧跟物联网技术发展趋势的高素质人才,高校物联网专业教学急需构建一套创新实践教学体系。本毕业设计旨在探索和设计一套新型…

社交创新的标杆:解读Facebook的社交模式

引言 在当今数字化时代,社交媒体已成为人们日常生活和沟通的重要工具。作为全球最大的社交媒体平台,Facebook不仅改变了我们的社交模式,而且对全球的社交文化、商业活动和公共事务产生了深远的影响。本文将深入探讨Facebook的社交模式&#…

yolov8目标检测 部署瑞芯微rk3588记录

1. 前置条件 本地电脑系统,ubuntu20.04 训练代码: 训练代码下载的ultralytics官方代码 SHA:6a2fddfb46aea45dd26cb060157d22cf14cd8c64 训练代码仅做数据修改,类别修改,代码结构未做任何修改 需要准备的代码&#…

视频批量高效剪辑,支持将视频文件转换为音频文件,轻松掌握视频格式

在数字化时代,视频内容日益丰富,管理和编辑这些视频变得愈发重要。然而,传统的视频剪辑软件往往操作复杂,难以满足高效批量处理的需求。现在,一款全新的视频批量剪辑神器应运而生,它支持将视频文件一键转换…

怎么使用JMeter进行性能测试?

一、简介 JMeter是Apache软件基金会下的一款开源的性能测试工具,完全由Java开发。它专注于对我们应用程序进行负载测试和性能测量,最初设计用于web应用程序,现在已经扩展到其他测试功能,比如:FTP、Database和LDAP等。…

如何用Redis高效实现12306的复杂售票业务

12306的售票业务是一个复杂的系统,需要考虑高并发、高可用、数据一致性等问题。使用Redis作为缓存和持久化存储,可以提高系统的性能和可扩展性,以下是一些可能的实现方式: 1 票源信息缓存:将票源信息(如车次…

Pytest精通指南(14)Parametrize之indirect(间接参数)

文章目录 官方概念概念分析官方示例示例分析验证indirect为True但不指定fixture验证indirect为True但不存在fixture 官方概念 概念分析 在pytest的pytest.mark.parametrize装饰器中,indirect参数用于指示是否应该从fixtures中解析参数值,而不是直接使用提…