Python迭代器iterator生成器generator使用解析


Posted in Python onOctober 24, 2019

1. 迭代

根据记录的前面的元素的位置信息 去访问后续的元素的过程 -遍历 迭代

2. 可迭代对象 iterable

如何判断可迭代对象的3种方式

  • 能够被迭代访问的对象 for in
  • 常用可迭代对象-list tuple str
  • from collections import Iterable
  • isinstance(obj, Iterable)

3. 可迭代对象

可迭代对象通过__iter__方法提供一个 可以遍历对象中数据的工具-迭代器

iter(可迭代对象) 可以获取可迭代对象的迭代器

通过迭代器可以迭代访问 数据

next(迭代器) ===== 迭代器对象.__next__()

可迭代对象的本质 提供了一个迭代器(遍历可迭代对象中的数据)

如何获取可迭代对象中的迭代器 迭代器对象 = iter(可迭代对象)

如果通过迭代器访问可迭代对象中下一个元素 元素的值 = next(迭代器对象)

如果迭代器遍历完成 抛出 停止迭代-异常StopIteration

如果需要实现一个迭代器 就需要实现__next__()

4. 迭代器 iterator

-- 迭代器访问可迭代对象中数据 判断对象是否是迭代器类型

from collections import Iterator
isinstance(obj, Iterator)

自己实现

迭代器本身也是可迭代对象 __iter__() 提供迭代器(self)

下一个元素的值 = next(迭代器) =====> __next__()

实现一个可迭代对象

from collections import Iterable
from collections import Iterator
import time


class MylistIterator(object):
  """这是Mylist类型的对应迭代器类型 """
  def __init__(self,data):
    # 需要被便利的数据
    self.data = data
    # 保存用户访问的位置
    self.index = 0

  def __iter__(self):
    """python规定 迭代器是一种可迭代对象"""
    return self

  def __next__(self):
    """next(ml_iterator) 相当于调用迭代器对象的.__next__()"""
    if self.index < len(self.data):
      ret = self.data[self.index]
      self.index += 1
      return ret
    else:
      # 访问完成 应该抛出异常
      raise StopIteration

class Mylist(object):
  """可迭代对象"""
  def __init__(self):
    self.data = [1,2,3,4,5]

  def __iter__(self):
    """提供迭代器"""
    # 返回迭代器对象
    mliter = MylistIterator(self.data)
    return mliter

# ml是一个可迭代类型
ml = Mylist()

# 获取可迭代对象的 迭代器对象
ml_iter = iter(ml)
print(isinstance(ml_iter, Iterator))
for i in ml:
  print(i)
  time.sleep(1)

"""
1 可迭代对象的本质 提供了一个迭代器(遍历可迭代对象中的数据)

2 如何获取可迭代对象中的迭代器 迭代器对象 = iter(可迭代对象)
    实际上相当于 可迭代对象.__iter__()
3 如果通过迭代器访问可迭代对象中下一个元素 元素的值 = next(迭代器对象)








如果迭代器遍历完成 抛出 停止迭代-异常StopIteration
"""
print(isinstance(ml, Iterable))

用迭代器完成斐波那契数列(难点在next)

"""兔子队列 某一项的值是前两项的和
1 1 2 3 5 8
"""

class Fib(object):
  def __init__(self,n):
    """初始化操作"""
    # n代表数列的长度
    self.n = n

    # 下标记录
    self.index = 0

    self.number1 = 0
    self.number2 = 1

  def __iter__(self):
    return self

  def __next__(self):
    """next(迭代器)=== .__next__()"""
    if self.index < self.n:
      ret = self.number1
      self.number1,self.number2 = self.number2,self.number2+self.number1
      self.index += 1
      return ret
    else:
      raise StopIteration

# list() tuple()都可以接收迭代器 并且将遍历到的数据存储到集合中
print(list(Fib(10)))
#
# # 打印斐波那契数列的前10项的值
# # for i in Fib(10):
# #   print(i)
# # 1 通过iter函数获取可迭代对象 Iterable 的迭代器 iterator
# ml_iterator = iter(Fib(1000))
#
# # 2 在循环内部不断调用next(迭代器) 获取下一个元素的值
# # 3 如果迭代完成 会抛出一个停止迭代的异常StopIteration

5. 生成器 generator

生成器是一种特殊的迭代器 --- 是迭代器, 并且有自己的特点

1 创建生成器表达式 [] ----》 (x for x in range(100))

2 生成器函数

凡是有yield关键字的函数都不是普通函数了 而是生成器函数

# 列表推导式
lis=[x for x in range(10)]
print(lis)

# 生成器表达式 中括号变圆括号
data=(x for x in range(10))
print(data)
# 遍历data
for i in data:
  print(i)

结果:

[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
<generator object <genexpr> at 0x02AE7120>
1
3
5
7
9

6. yield关键字的作用

挂起当前函数 将后面表达式的值 返回到调用生成器的地方

接收数据 并唤醒当前函数 并且紧接着上次运行的地址继续执行

7. 唤醒生成器的两种方式

生成器.send("数据")

next(生成器) === 生成器.send(None)

在第一次调用生成器对象的是 必须使用next()

在后续的情况下 send和next可以混用

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
Python sys.path详细介绍
Oct 17 Python
python实现批量改文件名称的方法
May 25 Python
以一个投票程序的实例来讲解Python的Django框架使用
Feb 18 Python
对Python多线程读写文件加锁的实例详解
Jan 14 Python
PySide和PyQt加载ui文件的两种方法
Feb 27 Python
python实现远程控制电脑
May 23 Python
将Pytorch模型从CPU转换成GPU的实现方法
Aug 19 Python
Python requests设置代理的方法步骤
Feb 23 Python
matplotlib绘制鼠标的十字光标的实现(自定义方式,官方实例)
Jan 10 Python
python tkinter实现定时关机
Apr 21 Python
解决Django transaction进行事务管理踩过的坑
Apr 24 Python
只用Python就可以制作的简单词云
Jun 07 Python
Python 取numpy数组的某几行某几列方法
Oct 24 #Python
Django和Flask框架优缺点对比
Oct 24 #Python
python命令 -u参数用法解析
Oct 24 #Python
使用python制作游戏下载进度条的代码(程序说明见注释)
Oct 24 #Python
用Python解数独的方法示例
Oct 24 #Python
Python3 sys.argv[ ]用法详解
Oct 24 #Python
window7下的python2.7版本和python3.5版本的opencv-python安装过程
Oct 24 #Python
You might like
php.ini中date.timezone设置分析
2011/07/29 PHP
js调用activeX获取u盘序列号的代码
2011/11/21 Javascript
JavaScript不刷新实现浏览器的前进后退功能
2014/11/05 Javascript
js实现编辑div节点名称的方法
2014/12/17 Javascript
AngularJS基础 ng-cloak 指令简单示例
2016/08/01 Javascript
Vuejs第一篇之入门教程详解(单向绑定、双向绑定、列表渲染、响应函数)
2016/09/09 Javascript
js模式化窗口问题![window.dialogArguments]
2016/10/30 Javascript
js获取元素下的第一级子元素的方法(推荐)
2017/03/05 Javascript
使用javaScript实现鼠标拖拽事件
2020/04/03 Javascript
浅谈Node.js之异步流控制
2017/10/25 Javascript
[00:53]TI3正赛第三天 DK怒破A队不败金身 现场国旗飘扬热血激昂
2013/08/10 DOTA
Python中的pack和unpack的使用
2018/03/12 Python
python实现反转部分单向链表
2018/09/27 Python
如何运行.ipynb文件的图文讲解
2019/06/27 Python
Django logging配置及使用详解
2019/07/23 Python
Django models.py应用实现过程详解
2019/07/29 Python
基于python框架Scrapy爬取自己的博客内容过程详解
2019/08/05 Python
python Event事件、进程池与线程池、协程解析
2019/10/25 Python
python双向链表原理与实现方法详解
2019/12/03 Python
Python 实现平台类游戏添加跳跃功能
2020/03/27 Python
Python3 selenium 实现QQ群接龙自动化功能
2020/04/17 Python
pycharm第三方库安装失败的问题及解决经验分享
2020/05/09 Python
StubHub澳大利亚:购买或出售您的门票
2019/08/01 全球购物
利物浦足球俱乐部官方商店(美国):Liverpool FC US
2019/10/09 全球购物
2014厂务公开实施方案
2014/02/17 职场文书
摄影专业毕业生求职信
2014/08/05 职场文书
小学假期安全广播稿
2014/09/28 职场文书
邀请函范文
2015/02/02 职场文书
个人年终总结范文
2015/03/09 职场文书
审查起诉阶段律师意见书
2015/05/19 职场文书
任长霞观后感
2015/06/16 职场文书
邓小平文选读书笔记
2015/06/29 职场文书
2015年中秋寄语
2015/07/31 职场文书
日本读研:怎样写好一篇日本研究计划书?
2019/07/15 职场文书
Springboot如何同时装配两个相同类型数据库
2021/11/17 Java/Android
Elasticsearch6.2服务器升配后的bug(避坑指南)
2022/09/23 Servers