当前位置: 首页 > news >正文

python 迭代器生成器

目录

一、可迭代对象

1.1 判断是否为可迭代对象

二、迭代器

2.1 判断对象是否是一个迭代器

2.2 手写一个迭代器

2.3 迭代器应用场景

三、生成器

3.1 生成器介绍

3.2 使用yield 关键字 生成器,来实现迭代器

3.3 生成器(yield关键字)的运行逻辑


一、可迭代对象

可迭代对象,就是 可以使用for 循环遍历的对象。

比如 list 列表

 tuple 元组

set 集合

str 字符串

dic 字典

如果想要使用for来遍历,这个对象必需具有 __iter__ 方法

1.1 判断是否为可迭代对象

方法一:

from typing import Iterable
isinstance(obj,Iterable)

方法二:

是否有__iter__ 方法

print(hasattr(list,"__iter__"))
print(hasattr(str,"__iter__"))
print(hasattr(tuple,"__iter__"))
print(hasattr(dict,"__iter__"))
print(hasattr(set,"__iter__"))True
True
True
True
True

二、迭代器

1、迭代器首先是一个可迭代对象,拥有__iter__ 方法,返回self

2、还必须要有 __next__ 方法。__next__ 方法 用来输出下一个元素。如果没有下一个值,则抛出StopIterator 异常

3、迭代器,使用for操作时,得到的每一个元素,会自动的调用__next__方法,从而得到下一个元素,依次循环,这样就能遍历所有的元素了。for语句迭代会忽略异常。

4、迭代器迭代的时候,使用for,不要使用while

5、迭代器只能迭代一次。(因为对象里用于迭代计算的下标一直是增加的)

6、迭代器比list占用更小的内存。

2.1 判断对象是否是一个迭代器

迭代器同时拥有 __iter__ 方法 与 __next__ 方法。

方法一:

isinstance(obj,Iterator)

注意,这里是Iterator 不是Iterable

列表list,是一个可迭代对象,但不是一个迭代器。

from typing import Iterable, Iteratora = [1,2,3]
print(isinstance(a,Iterable))
# True
print(isinstance(a,Iterator))
# False

将 可迭代对象,转化为一个迭代器,使用iter方法.

from typing import Iterable, Iteratora = [1,2,3]
print(isinstance(a,Iterable))
print(isinstance(a,Iterator))b = iter(a)
print(isinstance(b,Iterator))# True

2.2 手写一个迭代器

# -*- coding:utf-8 -*-
# @Author: 喵酱
# @time: 2023 - 02 -18
# @File: my_iterator.pyclass MyIterator(object):def __init__(self, stop: int, start=0):self.stop = stopself.value = startdef __next__(self):"""返回下一个元素,没有则抛出StopIteration 异常"""if self.value < self.stop:self.value += 1else:raise StopIterationreturn self.valuedef __iter__(self):return selfif __name__ == '__main__':myIterator = MyIterator(5)print(myIterator.__next__())print(myIterator.__next__())print(myIterator.__next__())print(myIterator.__next__())print(myIterator.__next__())print(myIterator.__next__())

输出结果:

1
2
3
4
5
Traceback (most recent call last):File "/Users/zhaohui/PycharmProjects/MyTest/cekai/my_iterator.py", line 27, in <module>print(myIterator.__next__())File "/Users/zhaohui/PycharmProjects/MyTest/cekai/my_iterator.py", line 16, in __next__raise StopIteration
StopIteration

使用for遍历

    myIterator = MyIterator(5)for i in myIterator:print(i)打印结果:
1
2
3
4
5

2.3 迭代器应用场景

迭代器,占用内存,远远小于list。

需要节省内存的场景。

三、生成器

3.1 生成器介绍

生成器:为了快速、方便的创建一个迭代器。

生成器,就是一个使用了yield 关键字的迭代器。

3.2 使用yield 关键字 生成器,来实现迭代器

举例:

list = [ 1,2,3,4,5]

每个元素平方,再生成一个新的list

new_list = [1,4,9,16,25]

方法一:

使用list实现

list = [1, 2, 3]
list1 = []for i in list:list1.append(i * i)
print(list1)

方法二:

手动创建一个迭代器


class Square(object):def __init__(self, start, stop):self.value = startself.stop = stopdef __iter__(self):return selfdef __next__(self):if self.value > self.stop:raise StopIterationqueare_num = self.value * self.valueself.value += 1return queare_numif __name__ == '__main__':square = Square(1, 5)for i,s in enumerate(square):print(i,s)

打印结果:

0 1
1 4
2 9
3 16
4 25

方法三 使用yield 关键字,实现一个生成器


def square_yield(start:int,stop:int):for i in range(start,stop+1):yield i*iif __name__ == '__main__':square_yield(1,4)

yield 就等价于迭代器中的 __next__ 方法

yield 关键字,与 return 方法比较


def square_yield(start:int,stop:int):for i in range(start,stop+1):yield i*idef squqre_return(start,stop):for i in range(start,stop+1):return i*iif __name__ == '__main__':square_yield(1,4)print(squqre_return(1,4))

方法中,return 方法,就表示这个函数结束了。

yield 返回的是一个 生成器类。

3.3 生成器(yield关键字)的运行逻辑

1、yield 关键字,与return 一样,会返回一个结果。 但是return返回的是 运行结果

但是yield返回的,可不是运行结果,返回的是一个迭代器的对象。只有这个对象调用__next__

的时候,才会执行业务逻辑

2、 yield 会记录每次的运行位置。

3、只有生成器的对象在调用__next__方法时,才会执行业务代码。业务代码执行完,函数就会变成非runnning状态(挂起状态)。直到下一次__next__方法被调用

举几个例子,方便我们理解一下

举例一:

------没有使用yield关键字

def f1():print("开始执行")return "f1被执行了"if __name__ == '__main__':fun = f1()

当调用f1函数时,f1里面的业务代码被执行了。

------使用yield关键字

def f():print("yield 1------------")a = 1yield aprint("yield 2=================")a = 2yield aprint("yield 3=================")a = 3yield aif __name__ == '__main__':generator = f()

运行结果为空,因为generator = f(),得到的是一个生成器的对象,没有调用业务代码。

当我们第一次调用__next__时

    generator = f()print(generator.__next__())

输出结果:

yield 1------------
1

当第二次调用__next__时

输出结果:

yield 2=================
2

 当第三次调用__next__时

输出结果:

yield 3=================
3

当生成器的对象,调用__next__方法时,yield 会记录每一次 运行的位置,当 下一次调用__next__方法时,业务代码会接着上一次的位置继续运行。

如下图:

 

整体代码:

# -*- coding:utf-8 -*-
# @Author: 喵酱
# @time: 2023 - 02 -18
# @File: diedai5.pydef f():print("yield 1------------")a = 1yield aprint("yield 2=================")a = 2yield aprint("yield 3=================")a = 3yield aif __name__ == '__main__':generator = f()print(generator.__next__())print(generator.__next__())print(generator.__next__())

举例二:

解释一下上面求平方的函数,的生成器

def square_yield(start:int,stop:int):for i in range(start,stop+1):yield i*i

每次__next__ 都会记录 业务代码运行位置。

--------当第一次调用__next__ 时:

i = 1 ,返回 1x1 的结果。内存中记录 i = 1

--------第二次调用__next__ 时:

上一次的记录是i=1,开启for下一轮的循环,则i=2.返回 2x2 的结果。内存中记录 i = 2

 

相关文章:

python 迭代器生成器

目录 一、可迭代对象 1.1 判断是否为可迭代对象 二、迭代器 2.1 判断对象是否是一个迭代器 2.2 手写一个迭代器 2.3 迭代器应用场景 三、生成器 3.1 生成器介绍 3.2 使用yield 关键字 生成器&#xff0c;来实现迭代器 3.3 生成器&#xff08;yield关键字&#xff09;…...

Iceberg基于Spark MergeInto语法实现数据的增量写入

SPARK SQL 基本语法 示例SQL如下 MERGE INTO target_table t USING source_table s ON s.id t.id //这里是JOIN的关联条件 WHEN MATCHED AND s.opType delete THEN DELETE // WHEN条件是对当前行进行打标的匹配条件 WHEN MATCHED AND s.opType update THEN…...

JavaScript Array(数组) 对象

JavaScript 中的 Array&#xff08;数组&#xff09;对象是一种用来存储一系列值的容器&#xff0c;它可以包含任意类型的数据&#xff0c;包括数字、字符串、对象等等。通过使用数组对象&#xff0c;我们可以轻松地组织和处理数据&#xff0c;以及进行各种操作&#xff0c;比如…...

Debian如何更换apt源

中科大 deb https://mirrors.ustc.edu.cn/debian/ stretch main non-free contrib deb https://mirrors.ustc.edu.cn/debian/ stretch-updates main non-free contrib deb https://mirrors.ustc.edu.cn/debian/ stretch-backports main non-free contrib deb-src https://mirr…...

Connext DDSPersistence Service持久性服务

DDS持久性服务,它保存了DDS数据样本,以便即使发布应用程序已经终止,也可以稍后将其发送到加入系统的订阅应用程序。 简介Persistence Service是一个Connext DDS应用程序,它将DDS数据样本保存到临时或永久存储中,因此即使发布应用程序已经终止,也可以稍后将其交付给加入系…...

自抗扰控制ADRC之微分器TD

目录 前言 1 全程快速微分器 1.1仿真分析 1.2仿真模型 1.3仿真结果 1.4结论 2 Levant微分器 2.1仿真分析 2.2仿真模型 2.3仿真结果 3.总结 前言 工程上信号的微分是难以得到的&#xff0c;所以本文采用微分器实现带有噪声的信号及其微分信号提取&#xff0c;从而实现…...

链表学习之复制含随机指针的链表

链表解题技巧 额外的数据结构&#xff08;哈希表&#xff09;&#xff1b;快慢指针&#xff1b;虚拟头节点&#xff1b; 复制含随机指针的链表 该链表节点的结构如下&#xff1a; class ListRandomNode { public:ListRandomNode() : val(0), next(nullptr), random(nullptr…...

【人脸检测】Yolov5Face:优秀的one-stage人脸检测算法

论文题目&#xff1a;《YOLO5Face: Why Reinventing a Face Detector》 论文地址&#xff1a;https://arxiv.org/pdf/2105.12931.pdf 代码地址&#xff1a;https://github.com/deepcam-cn/yolov5-face 1.简介 近年来&#xff0c;CNN在人脸检测方面已经得到广泛的应用。但是许多…...

【Unity3d】Unity与Android之间通信

在unity开发或者sdk开发经常遇到unity与移动端原生层之间进行通信&#xff0c;这里把它们之间通信做一个整理。 关于Unity与iOS之间通信&#xff0c;参考【Unity3d】Unity与iOS之间通信 Unity(c#)调用Android (一)、编写Java代码 实际上&#xff0c;任何已经存在的Java代码…...

Allegro如何更改DRC尺寸大小操作指导

Allegro如何更改DRC尺寸大小操作指导 在做PCB设计的时候,DRC可以辅助设计,有的时候DRC的尺寸过大会影响视觉,Allegro支持将DRC的尺寸变小或者改大 如下图,DRC尺寸过大 如何改小,具体操作如下 点击Setup选择Design Parameters...

Mongodb WT_PANIC: WiredTiger library panic

文章目录故障现象排查过程1.查看Log2.同步恢复数据故障现象 周五突然收到Mongo实例莫名奇妙挂了告警&#xff0c;一般都是RS复制集架构模式&#xff08;5节点&#xff09;&#xff0c;查看此实例角色为SECONDAR&#xff0c;挂了暂时不影响线上业务&#xff0c;但还是需要尽快修…...

【HTML】HTML 表格总结 ★★★ ( 表格标签 | 行标签 | 单元格标签 | 表格标签属性 | 表头单元格标签 | 表格标题标签 | 合并单元格 )

文章目录一、表格标签组成 ( 表格标签 | 行标签 | 单元格标签 )二、table 表格属性 ( border 属性 | align 属性 | width 属性 | height 属性 )三、表头单元格标签四、表格标题标签五、合并单元格1、合并单元格方式2、合并单元格顺序3、合并单元格流程六、合并单元格示例1、原始…...

linux013之文件和目录的权限管理

用户、组、文件目录的关系&#xff1a; 简介&#xff1a;用户和组关联&#xff0c;组合文件目录关联&#xff0c;这样就实现了用户对文件的权限管理。首先来看一下&#xff0c;一个文件或目录的权限是怎么查看的&#xff0c;ls -l&#xff0c; 如下&#xff0c;这个信息怎么看呢…...

设计模式之状态模式

什么是状态模式 状态模式是指允许一个对象在其内部状态改变时改变他的行为&#xff0c;对象看起来似乎改变了整个类。     状态模式将一个对象在不同状态下的不同行为封装在一个个状态类中&#xff0c;通过设置不同的状态对象可以让环境对象拥有不同的行为&#xff0c;而状…...

XQuery 选择 和 过滤

XML实例文档 我们将在下面的例子中继续使用这个 "books.xml" 文档&#xff08;和上面的章节所使用的 XML 文件相同&#xff09;。 在您的浏览器中查看 "books.xml" 文件。 选择和过滤元素 正如在前面的章节所看到的&#xff0c;我们使用路径表达式或 FL…...

室友打了一把王者的时间,我理清楚了grep,find,管道|,xargs的区别与联系,用的时候不知道为什么要这样用

目录 问题引入 find和grep的基本区别 xargs命令 Linux命令的标准输入 vs 命令行参数 举例总结 问题引入 在自己做项目的过程中&#xff0c;想使用linux命令统计下一个目录下html文件的数量&#xff0c;在思考应该使用grep还是find去配合wc指令统计文件数量&#xff0c;后来…...

python 刷题时常见的函数

collections.OrderedDict 1. move_to_end() move_to_end() 函数可以将指定的键值对移动到最前面或者最后面&#xff0c;即最左边或最右边 。 2. popitem() popitem()可以完成元素的删除操作&#xff0c;有一个可选参数last&#xff08;默认为True&#xff09;&#xff0c;…...

Python之列表推导式和列表排序

Python中的列表推导式&#xff0c;是小编比较喜欢的一种&#xff0c;他能大大减少你的代码量来得到你想要的结果&#xff0c;下面说说列表中常用的几种推导式 列表排序 Python开发中会经常用到排序操作&#xff0c;这里提供两种方式供大家参考&#xff0c;对象的sort()方法和…...

力扣(LeetCode)240. 搜索二维矩阵 II(C++)

题目描述 枚举 枚举整个矩阵&#xff0c;找到等于 target 的元素&#xff0c;则 return true &#xff0c;否则 return false。 class Solution { public:bool searchMatrix(vector<vector<int>>& matrix, int target) {int n matrix.size(), m matrix[0]…...

golang defer

文章目录延迟函数的参数在defer语句出现时就已经确定下来了延迟函数没有入参时&#xff0c;延迟函数体内的变量会受到影响延迟函数 *可以* 修改主函数的 *具名* 返回值延迟函数 *无法* 修改主函数的 *匿名* 返回值defer会把声明的 延迟函数以及 函数的入参放到栈上&#xff0c;…...

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…...

工程地质软件市场:发展现状、趋势与策略建议

一、引言 在工程建设领域&#xff0c;准确把握地质条件是确保项目顺利推进和安全运营的关键。工程地质软件作为处理、分析、模拟和展示工程地质数据的重要工具&#xff0c;正发挥着日益重要的作用。它凭借强大的数据处理能力、三维建模功能、空间分析工具和可视化展示手段&…...

css的定位(position)详解:相对定位 绝对定位 固定定位

在 CSS 中&#xff0c;元素的定位通过 position 属性控制&#xff0c;共有 5 种定位模式&#xff1a;static&#xff08;静态定位&#xff09;、relative&#xff08;相对定位&#xff09;、absolute&#xff08;绝对定位&#xff09;、fixed&#xff08;固定定位&#xff09;和…...

iOS性能调优实战:借助克魔(KeyMob)与常用工具深度洞察App瓶颈

在日常iOS开发过程中&#xff0c;性能问题往往是最令人头疼的一类Bug。尤其是在App上线前的压测阶段或是处理用户反馈的高发期&#xff0c;开发者往往需要面对卡顿、崩溃、能耗异常、日志混乱等一系列问题。这些问题表面上看似偶发&#xff0c;但背后往往隐藏着系统资源调度不当…...

CVE-2020-17519源码分析与漏洞复现(Flink 任意文件读取)

漏洞概览 漏洞名称&#xff1a;Apache Flink REST API 任意文件读取漏洞CVE编号&#xff1a;CVE-2020-17519CVSS评分&#xff1a;7.5影响版本&#xff1a;Apache Flink 1.11.0、1.11.1、1.11.2修复版本&#xff1a;≥ 1.11.3 或 ≥ 1.12.0漏洞类型&#xff1a;路径遍历&#x…...

push [特殊字符] present

push &#x1f19a; present 前言present和dismiss特点代码演示 push和pop特点代码演示 前言 在 iOS 开发中&#xff0c;push 和 present 是两种不同的视图控制器切换方式&#xff0c;它们有着显著的区别。 present和dismiss 特点 在当前控制器上方新建视图层级需要手动调用…...

python爬虫——气象数据爬取

一、导入库与全局配置 python 运行 import json import datetime import time import requests from sqlalchemy import create_engine import csv import pandas as pd作用&#xff1a; 引入数据解析、网络请求、时间处理、数据库操作等所需库。requests&#xff1a;发送 …...

在树莓派上添加音频输入设备的几种方法

在树莓派上添加音频输入设备可以通过以下步骤完成&#xff0c;具体方法取决于设备类型&#xff08;如USB麦克风、3.5mm接口麦克风或HDMI音频输入&#xff09;。以下是详细指南&#xff1a; 1. 连接音频输入设备 USB麦克风/声卡&#xff1a;直接插入树莓派的USB接口。3.5mm麦克…...

消息队列系统设计与实践全解析

文章目录 &#x1f680; 消息队列系统设计与实践全解析&#x1f50d; 一、消息队列选型1.1 业务场景匹配矩阵1.2 吞吐量/延迟/可靠性权衡&#x1f4a1; 权衡决策框架 1.3 运维复杂度评估&#x1f527; 运维成本降低策略 &#x1f3d7;️ 二、典型架构设计2.1 分布式事务最终一致…...

使用SSE解决获取状态不一致问题

使用SSE解决获取状态不一致问题 1. 问题描述2. SSE介绍2.1 SSE 的工作原理2.2 SSE 的事件格式规范2.3 SSE与其他技术对比2.4 SSE 的优缺点 3. 实战代码 1. 问题描述 目前做的一个功能是上传多个文件&#xff0c;这个上传文件是整体功能的一部分&#xff0c;文件在上传的过程中…...