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+Turtle动态绘制一棵树实例分享
Jan 16 Python
读取json格式为DataFrame(可转为.csv)的实例讲解
Jun 05 Python
Python读取csv文件分隔符设置方法
Jan 14 Python
pycharm+PyQt5+python最新开发环境配置(踩坑)
Feb 11 Python
Python实现简单查找最长子串功能示例
Feb 26 Python
Python中函数的返回值示例浅析
Aug 28 Python
浅谈python已知元素,获取元素索引(numpy,pandas)
Nov 26 Python
python Qt5实现窗体跟踪鼠标移动
Dec 13 Python
pytorch 利用lstm做mnist手写数字识别分类的实例
Jan 10 Python
python argparse传入布尔参数false不生效的解决
Apr 20 Python
Django models文件模型变更错误解决
May 11 Python
Django celery异步任务实现代码示例
Nov 26 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实现异步调用方法研究与分享
2011/10/27 PHP
PHP使用DirectoryIterator显示下拉文件列表的方法
2015/03/13 PHP
CodeIgniter控制器之业务逻辑实例分析
2016/01/20 PHP
Adnroid 微信内置浏览器清除缓存
2016/07/11 PHP
ajax 缓存 问题 requestheader
2010/08/01 Javascript
javascript new fun的执行过程
2010/08/05 Javascript
javascript实现类似超链接的效果
2014/12/26 Javascript
jQuery鼠标事件汇总
2015/08/30 Javascript
Vue动态实现评分效果
2017/05/24 Javascript
详解webpack+vue-cli项目打包技巧
2017/06/17 Javascript
浅谈mint-ui loadmore组件注意的问题
2017/11/08 Javascript
基于vue.js实现分页查询功能
2018/12/29 Javascript
微信小程序使用map组件实现获取定位城市天气或者指定城市天气数据功能
2019/01/22 Javascript
cordova+vue+webapp使用html5获取地理位置的方法
2019/07/06 Javascript
JavaScript实现飞舞的泡泡效果
2020/02/07 Javascript
vue动态合并单元格并添加小计合计功能示例
2020/11/26 Vue.js
[13:18]《一刀刀一天》之DOTA全时刻21:详解TI新赛制 A队再露獠牙
2014/06/24 DOTA
python远程登录代码
2008/04/29 Python
Python聚类算法之凝聚层次聚类实例分析
2015/11/20 Python
分享一个简单的python读写文件脚本
2017/11/25 Python
解决Python网页爬虫之中文乱码问题
2018/05/11 Python
python 定时任务去检测服务器端口是否通的实例
2019/01/26 Python
详解有关PyCharm安装库失败的问题的解决方法
2020/02/02 Python
pytorch对梯度进行可视化进行梯度检查教程
2020/02/04 Python
使用Keras实现简单线性回归模型操作
2020/06/12 Python
Python子进程subpocess原理及用法解析
2020/07/16 Python
Timberland美国官网:全球领先的户外品牌
2016/08/15 全球购物
什么是Smarty变量操作符?如何使用Smarty变量操作符
2014/07/18 面试题
小型女装店的创业计划书
2014/01/09 职场文书
优秀纪检干部材料
2014/08/27 职场文书
党员学习正风肃纪思想汇报
2014/09/12 职场文书
投标人法定代表人授权委托书格式
2014/09/28 职场文书
2015大学生实训报告
2014/11/05 职场文书
申报优秀教师材料
2014/12/16 职场文书
初中生思想道德自我评价
2015/03/09 职场文书
在pyCharm中下载第三方库的方法
2021/04/18 Python