Python教程(26)——Python迭代器和生成器详解

news2024/12/28 3:16:20

迭代器

Python中的迭代器是一种对象,它可以迭代(遍历)一个可迭代对象(比如列表、元组或字符串)的元素。迭代器用于实现迭代器协议,即包含 __iter__() 方法和 __next__() 方法。

迭代器的工作原理是每次调用 __next__() 方法时返回可迭代对象的下一个元素,当没有元素可迭代时,抛出 StopIteration 异常。

class MyIterator:
    def __init__(self, iterable):
        self.iterable = iterable
        self.index = 0

    def __iter__(self):
        return self

    def __next__(self):
        if self.index < len(self.iterable):
            result = self.iterable[self.index]
            self.index +=1
            return result
        else:
            raise StopIteration

# 创建一个可迭代对象
my_list = [1, 2, 3, 4, 5]

# 创建一个迭代器
my_iterator = MyIterator(my_list)

# 使用迭代器遍历元素
for item in my_iterator:
    print(item)

值得注意的是,当迭代器耗尽后,如果再次使用迭代器来便利,将不会得到任何输出。
所以总的来说,迭代器是用于遍历可迭代对象的对象,它实现了迭代器协议,具有 __iter__()__next__() 方法。

可迭代对象

我们看迭代器好像和平时我们使用的列表、字典等数据结构一样,都可以遍历,那么列表、字典等数据结构是迭代器吗?

不好意思,他们不是迭代器,而是可迭代对象(iterable)。

可迭代对象(iterable)是指具有迭代行为的对象。当我们希望能够按照一定方式遍历对象中的元素时,我们可以将该对象称为可迭代对象。换句话说,可迭代对象是一种提供迭代能力的容器。
可迭代对象的特点是实现了 __iter__() 方法,这个方法返回一个迭代器(iterator)。迭代器是能够按照一定顺序生成下一个元素的对象。

在 Python 中,许多数据结构都是可迭代对象,比如列表、元组、集合、字典等。我们可以使用for循环对这些对象进行遍历。

同时,也可以使用内置的 iter() 函数将可迭代对象转换为迭代器。迭代器是可迭代对象的一种特殊形式,实现了 __iter__()__next__() 方法。迭代器可以使用 next() 函数来获取下一个元素,并且在没有元素可返回时引发 StopIteration 异常。

my_list = [1, 2, 3, 4, 5]  # 列表是可迭代对象
for item in my_list:
    print(item)

my_iterator = iter(my_list)  # 使用iter()函数将列表这个可迭代对象转换为迭代器
print(next(my_iterator))  # 输出第一个元素
print(next(my_iterator))  # 输出第二个元素

在这个示例中,列表 my_list 是可迭代对象,它可以被 for 循环遍历。另外,我们还使用 iter() 函数将 my_list 转换为迭代器 my_iterator,并使用 next() 函数逐个访问其中的元素。

所以总的来说,可迭代对象是指具有迭行为的对象,它们实现了 __iter__() 方法。通过for循环或 iter() 函数,我们可以遍历这些对象的元素。

可迭代对象是指实现了 __iter__() 方法的对象,而迭代器是实现了 __iter__()__next__() 方法的对象,这个可以说是它们比较明显的区别。

for循环机制

从上面我们指定,列表、元组、集合、字典等数据结构是可迭代对象,并不是迭代器。而可迭代对象只实现了__iter__()方法,并不具有迭代(也就是返回下一个元素)的功能。那么很多同学可能就比较奇怪了,我们平时使用for循环遍历这个数据结构的时候,内部是怎么遍历的呢?

实际上在 Python 中,for循环在内部自动会调用 __iter__() 函数将可迭代对象转换为迭代器。用for循环遍历可迭代对象的实现机制为:

  • for循环首先会调用 __iter__() 函数,该函数会将可迭代对象转换为一个迭代器对象(如果对象本身就是迭代器,则不作转换)。
  • 接下来,for循环会调用迭代器对象的 __next__() 方法来获取下一个元素。如果迭代器对象没有下一个元素,会抛出 StopIteration 异常。
  • for循环会自动捕捉 StopIteration 异常,表示已经迭代完所有元素,循环将结束。

所以以下两个方法实际上是等价的。

my_string = "Hello"

for char in my_string:
    print(char)

实际上完全等价于:

my_string = "Hello"
my_iterator = iter(my_string)

print(next(my_iterator))  # 输出:H
print(next(my_iterator))  # 输出:e
print(next(my_iterator))  # 输出:l
print(next(my_iterator))  # 输出:l
print(next(my_iterator))  # 输出:o

生成器

生成器(Generator)是一种特殊的迭代器,它可以在迭代过程中动态地生成值,而不是一次性地将所有元素放在内存中。生成器使用 yield 关键字来定义,当生成器的代码块执行到 yield 语句时,就会暂停执行并返回一个值,下次调用时会从上次暂停的位置继续执行。这样可以在需要的时候生成值,而不是一次性生成所有的值。

def my_generator():
    yield 1
    yield 2
    yield 3

# 使用生成器
gen = my_generator()

print(next(gen))  # 输出:1
print(next(gen))  # 输出:2
print(next(gen))  # 输出:3
print(next(gen))  # 抛出 StopIteration 异常

在上面的示例中,我们定义了一个名为 my_generator 的生成器函数,它通过使用 yield 关键字来产生值。当我们调用生成器函数时,它返回一个生成器对象 gen。我们可以使用 next() 函数来逐个获取生成器的值。每次调用 next() 时,生成器函数会从上次暂停的位置继续执行,并返回 yield 语句的值。当生成器函数执行完毕或没有更多的值可生成时,调用 next() 会抛出 StopIteration 异常。

生成器的一个重要特点是它们可以节省内存,尤其在处理大量数据时非常有用。由于生成器是按需生成值,只有在需要时才会在迭代过程中生成值,不会一次性占用大量的内存。所在遍历大批量数据的时候,非常有用,因为如果将大批量的数据直接加载到内存中在遍历,肯定会消耗很多内存,而利用生成器就可以做到需要哪些遍历哪些。

如果用简单一句话来说就是,我既想大量的数据,又想让它占用空间少,实现鱼和熊掌的兼得,那么就用生成器!

def read_large_file(file_path):
    with open(file_path, 'r') as file:
        for line in file:
            yield line.rstrip()

# 使用生成器遍历大文件
file_generator = read_large_file('large_file.txt')
print(file_generator) # 输出<generator object fibonacci_generator at 0x0000017362DCFED0>
print(type(file_generator)) # 输出<class 'generator'>
for line in file_generator:
    # 处理每一行数据
    print(line)

从上面可以看出,read_large_file() 函数是一个生成器函数,它按行从一个大文件中读取数据。通过使用 yield 关键字,在每次迭代时逐行生成文件的内容,并将其作为生成器的值返回。然后,我们可以使用 for 循环逐行处理大文件。可以看到,我们打印的file_generator类型是一个生成器。

当然,以上的例子并不一定需要采用生成器才能处理,我们直接在第4行进行处理也是可以的,生成器更多的只是提供一种思路,当你用常规方法不能解决问题的时候,可以试试用生成器。

生成器原理

生成器的原理基于迭代器(iterators)和生成器函数(generator functions)。

def even_numbers(n):
    for i in range(1, n+1):
        if i % 2 == 0:
            yield i

# 创建生成器对象
even_generator = even_numbers(10)

# 打印生成的偶数
for number in even_generator:
    print(number)

生成器函数

生成器通过生成器函数创建,生成器函数是一种特殊类型的函数,使用 yield 语句来生成值。像上面的例子even_numbers函数就是要给生成器函数,当调用生成器函数时,它返回一个生成器对象,而不是立即执行函数体内的代码,even_generator就是一个生成器对象。

而之前我们说过生成器是一种特殊类型的迭代器,它可以在迭代中生成值。迭代器是一个实现了 __iter__()__next__() 方法的对象。__iter__() 方法返回迭代器本身,而for循环内部会自动调用 __next__() 方法用于获取下一个值。每次调用 __next__() 方法时,生成器会从上一次暂停的位置继续执行,直到遇到下一个 yield 语句,然后将 yield 后面的值返回给调用者。

逐个生成值

生成器在调用 __next__() 方法时逐个生成值,并且每次在生成一个值后会暂停执行。这种延迟生成的机制使得生成器能够处理大量数据或无限序列,而不需要一次性加载或计算所有值。

状态保存

生成器在暂停执行时会保存其状态,包括局部变量、指令指针等信息。下一次调用 __next__() 方法时,生成器会从上一次暂停的地方恢复执行,并继续执行剩余部分的代码。

更多精彩内容,请关注同名公众:一点sir(alittle-sir)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1452709.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

2.11题目

#include <stdio.h> int main() { char a; while((a getchar()) ! -1) { if(a > A && a < Z) a32; putchar(ch); } return 0;} ———————————————— 版权声明&#xff1a;本文为博主原创文章…

2024年危险化学品经营单位主要负责人证模拟考试题库及危险化学品经营单位主要负责人理论考试试题

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 2024年危险化学品经营单位主要负责人证模拟考试题库及危险化学品经营单位主要负责人理论考试试题是由安全生产模拟考试一点通提供&#xff0c;危险化学品经营单位主要负责人证模拟考试题库是根据危险化学品经营单位主…

【大模型 数据增强】LLMAAA:使用 LLMs 作为数据标注器

【大模型 数据增强】LLMAAA&#xff1a;使用 LLMs 作为数据标注器 提出背景算法步骤1. LLM作为活跃标注者&#xff08;LLMAAA&#xff09;2. k-NN示例检索与标签表述化3. 活跃学习策略4. 自动重权技术 LLMAAA 框架1. LLM Annotator2. Active Acquisition3. Robust Training 总结…

数据结构对链表的初步认识(一)

已经两天没有更新了&#xff0c;今天就写一篇数据结构的链表吧&#xff0c;巩固自己也传授知识&#xff0c;不知道各位是否感兴趣看看这一篇有关联表的文章。 目录 链表的概念与结构 单向链表的实现 链表各个功能函数 首先我在一周前发布了一篇有关顺序表的文章&#xff0c;…

【Linux系统化学习】缓冲区

目录 缓冲区 一个样例 现象解释 缓冲区存在的位置 缓冲区 在刚开始学习C语言的时候我们就听过缓冲区这个名词&#xff0c;很是晦涩难懂&#xff1b;在Linux下进程退出时也包含缓冲区&#xff0c;因此缓冲区到底是什么&#xff1f;有什么作用&#xff1f; 让我们先从一个小…

斯坦福大学全能家政服务机器人Mobile ALOHA以及“小群体大智慧”Zooids集群机器人

斯坦福大学成功研发出低成本自主进化克隆人类行为和任务的能力全能型家政服务机器人。 原文标题: 【Mobile ALOHA-Learning Bimanual Mobile Manipulation with Low-Cost Whole-Body Teleoperation】 论文链接:【Mobile ALOHA (mobile-aloha.github.io)】。 以及由斯坦福大学…

51单片机项目(32)——基于51单片机的温度检测及控制装置的proteus仿真

1.功能设定 使用DS18B20测定当前温度并实时显示在LCD1602屏幕&#xff0c;使用四个按键设定温度的上限、下限。当温度低于下限时&#xff0c;蜂鸣器报警同时开启升温装置&#xff1b;当温度大于上限时&#xff0c;蜂鸣器报警同时启动降温装置。 仿真图如下&#xff1a; 2.软件设…

【无标题】管理kvm 虚拟机

管理kvm 虚拟机 点击虚拟机 创建新的虚拟机 安装操作系统 设置root密码

中小学信息学奥赛CSP-J认证 CCF非专业级别软件能力认证-入门组初赛模拟题第三套(选择题)

CSP-J入门组初赛模拟练习题第三套 1、以下不是属于国家顶级域名的是 A、.au B、.cn C、.com D、.jp 答案&#xff1a;C 考点分析&#xff1a;主要考查域名相关知识&#xff0c;au是澳大利亚、cn是中国&#xff0c;jp是日本&#xff0c;答案C 2、2个10进制数1111和1010的异…

蓝桥杯 星期计算

思路1 由于2022太大&#xff0c;用double来存储&#xff0c;即(52022 % 7) % 7即可 int num 5;int t (int)(Math.pow(20,22)%7);num t;num%7;System.out.println(num1);思路2 你需要知道 (a * b ) % p a % p * b % p Scanner scan new Scanner(System.in);int num 1;for…

计算机设计大赛 深度学习YOLO抽烟行为检测 - python opencv

文章目录 1 前言1 课题背景2 实现效果3 Yolov5算法3.1 简介3.2 相关技术 4 数据集处理及实验5 部分核心代码6 最后 1 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 &#x1f6a9; 基于深度学习YOLO抽烟行为检测 该项目较为新颖&#xff0c;适合作为竞赛课…

【感知机】感知机(perceptron)学习策略

感知机( perceptron )是二类分类的线性分类模型&#xff0c;其输入为实例的特征向量&#xff0c;输出为实例的类别&#xff0c;取1 和-1二值。感知机对应输入空间(特征空间)中将实例划分为正负两类的分离超平面&#xff0c;是一种判别模型。感知机是神经网络与支持向量机的基础…

Linux系统——拓展LVM逻辑卷分区与磁盘配额

一、LVM逻辑卷分区 1.检测并确认新硬盘 1.1fdisk 查看或管理硬盘分区 fdisk -l&#xff08;小写的L&#xff09; &#xff08;硬盘设备&#xff09; 或 fdisk 硬盘设备 1.2实际操作 1.2.1fdisk查询结果详解 Device&#xff1a;分区的设备文件名称Boot&#xff1a;是否…

WordPress主题YIA在广告位添加图片广告时下方有空白怎么办?

YIA主题设置中默认有4个广告位&#xff0c;而侧边栏的广告位由站长自行添加。boke112百科在这些广告位添加图片广告后发现图片下方有空白&#xff0c;导致下方的两个角没有变圆角&#xff0c;看起来也有点不好看。具体如下图所示&#xff1a; 其实&#xff0c;这个问题就是典型…

PyTorch-线性回归

已经进入大模微调的时代&#xff0c;但是学习pytorch&#xff0c;对后续学习rasa框架有一定帮助吧。 <!-- 给出一系列的点作为线性回归的数据&#xff0c;使用numpy来存储这些点。 --> x_train np.array([[3.3], [4.4], [5.5], [6.71], [6.93], [4.168],[9.779], [6.1…

【开源图床】使用Typora+PicGo+Github+CDN搭建个人博客图床

准备工作&#xff1a; 首先电脑得提前完成安装如下&#xff1a; 1. nodejs环境(node ,npm):【安装指南】nodejs下载、安装与配置详细教程 2. Picgo:【安装指南】图床神器之Picgo下载、安装与配置详细教程 3. Typora:【安装指南】markdown神器之Typora下载、安装与无限使用详细教…

模型 4i(趣味、利益、互动、个性)理论

系列文章 分享 模型&#xff0c;了解更多&#x1f449; 模型_总纲目录。重在提升认知。以用户为中心营销。 1 模型 4i(趣味、利益、互动、个性)理论的应用 1.1 4i理论在电子商务中的应用 小米公司在其电子商务平台上运用了 4i理论&#xff0c;取得了较好的效果。具体表现如下…

Netty Review - ByteBuf扩容机制源码解析

文章目录 Pre概述前置知识&#xff1a; 名词解释writeByte 源码解析实现ensureWritable0(minWritableBytes)ensureWritable0alloc().calculateNewCapacity 总结 Pre Netty Review - 直接内存的应用及源码分析 Netty Review - 底层零拷贝源码解析 Netty Review - ByteBuf内存…

《Go 简易速速上手小册》第3章:数据结构(2024 最新版)

文章目录 3.1 数组与切片&#xff1a;Go 语言的动态队伍3.1.1 基础知识讲解3.1.2 重点案例&#xff1a;动态成绩单功能描述实现代码扩展功能 3.1.3 拓展案例 1&#xff1a;数据分析功能描述实现代码扩展功能 3.1.4 拓展案例 2&#xff1a;日志过滤器功能描述实现代码扩展功能 3…

LeetCode-第70题-爬楼梯

1.题目描述 假设你正在爬楼梯。需要 n 阶你才能到达楼顶。 每次你可以爬 1 或 2 个台阶。你有多少种不同的方法可以爬到楼顶呢&#xff1f; 2.样例描述 3.思路描述 画图就可以发现规律&#xff0c;典型的斐波那契额数列 4.代码展示 class Solution {public int climbStair…