一.可迭代对象
对象必须提供一个__iter__()方法,如果有,那么就是可迭代对象,
像列表,元祖,字典等都是可迭代对象
可使用isinstance(obj,Iterable)方法判断
 from collections import Iterable,Iterator
l={'':2,'fd':5,'f':6}
l_i=l.__iter__() print(isinstance(l,Iterable))
print(isinstance(l_i,Iterator))

结果:

True

True

二.迭代器

迭代器协议:
对象必须提供一个next方法,执行该方法要么返回迭代中的下一项,要么就引起一个StopIteration异常,以终止迭代 (只能往后走不能往前退)
符合迭代器协议的就是迭代器。
一个对象是迭代器也是可迭代对象
from collections import Iterable,Iterator
l={'2':2,'fd':5,'f':6}
l_i=l.__iter__()
s=isinstance(l,Iterable)
print(s)
print(isinstance(l_i,Iterator))
打印结果:
True
True
三.生成器
遵循迭代器协议,相当于一种特殊的迭代器对象
1.生成器表达式
(1)三元表达式(必须是三元)
name = '李'
res='儿子' if name == '李' else '爸爸'
print(res)
打印结果:
儿子
(2)列表解析
所谓列表解析就是列表内包含两元或三元表达式(不能是四元)
 l=['蚂蚁%s' %i for i in range(10)]
l1=['蚂蚁%s' %i for i in range(10) if i >5]
print(l)
print(l1)
打印结果:
['蚂蚁0', '蚂蚁1', '蚂蚁2', '蚂蚁3', '蚂蚁4', '蚂蚁5', '蚂蚁6', '蚂蚁7', '蚂蚁8', '蚂蚁9']
['蚂蚁6', '蚂蚁7', '蚂蚁8', '蚂蚁9']
一行代码写出9*9乘法表:
print('\n'.join([' '.join(["%dX%d=%-2d" %(i,m,m*i) for i in range(1,m+1)]) for m in range(1,10)]))
原理:.join()方法处理的结果是一个整体的字符串
(3)生成器表达式
  l=['蚂蚁%s' %i for i in range(10)]#列表
l=('蚂蚁%s' %i for i in range(10))#生成器表达式
和列表相比不占用内存,使用一次生成一次
l=('蚂蚁%s' %i for i in range(10))
print(l)
print(l.__next__())
print(l.__next__())
print(next(l))
print(next(l))
运行结果:
<generator object <genexpr> at 0x000002191AFB5BA0>
蚂蚁0
蚂蚁1
蚂蚁2
蚂蚁3 2.生成器函数
优点:
不会立即执行,节省内存。
有需要时用一个现运行出一个,yield会保留当前状态,下次从当前位置继续运行,直到再次碰到yeild。
对比两个过程理解:
过程一(做出一个卖一个,即调用一次现运行出来一个结果)
 def product_baozi():
for i in range(100):
print('正在生产包子%s' %i)
yield '一屉包子%s' %i #i=1
print('开始卖包子')
pro_g=product_baozi()
baozi1=pro_g.__next__()
运行结果:
正在生产包子0
过程二:(调用第二次,现运行出来第二个结果)
 def product_baozi():
for i in range(100):
print('正在生产包子%s' %i)
yield '一屉包子%s' %i #i=1
print('开始卖包子')
pro_g=product_baozi() baozi1=pro_g.__next__()
baozi1=pro_g.__next__()
打印结果:
正在生产包子0
开始卖包子
正在生产包子1
3.生成器特性
特性一:for循环可以遍历生成器
def eges():
for i in range(5):
yield '鸡蛋%s' %i #i=1
pro_g=eges() for i in pro_g:
print(i)
特性二:生成器只能迭代一次,用完就没了
 with open('人口普查','r',encoding='utf-8') as f:
def g_population():
for i in f:
yield i print(sum(eval(l)['population'] for l in g_population()))
print(g_population().__next__()) #此句会抛出异常
4.利用生成器实现单线程并发
send()方法,
有一个参数,该参数指定的是上一次被挂起的yield语句的返回值,
并且再次遇到yeild会向send传回一个值。
实例:饭店吃饭
 import time
def consetomer():
print('服务员点餐')
time.sleep(5)
for i in range(5):
cai=yield i #接受send的传值,并向下执行,直到结束或遇到下一个yield(此时的yield会向send传回一个值)
print('顾客开始吃第%s个菜' %cai) def canting():
g1=consetomer()
print(g1.__next__())
print('厨师准备做菜')
for i in range(1,5):
time.sleep(1)
print('第%s个菜做好了' %i)
print('第%s个好吃' %g1.send(i)) #传送上一次挂起的yield并等待接受下一次yeild返回值
canting()

 

打印结果:

 服务员点餐
0
厨师准备做菜
第1个菜做好了
顾客开始吃第1个菜
第1个好吃
第2个菜做好了
顾客开始吃第2个菜
第2个好吃
第3个菜做好了
顾客开始吃第3个菜
第3个好吃
第4个菜做好了
顾客开始吃第4个菜
第4个好吃
 

python(可迭代对象,迭代器,生成器及send方法详解)的更多相关文章

  1. python 可迭代对象 迭代器 生成器总结

    可迭代对象 只要有魔法方法__iter__的就是可迭代对象  list和tuple和dict都是可迭代对象 迭代器 只要有魔法方法__iter__和__next__的就是可迭代对象 生成器 只要含有y ...

  2. Python学习之旅—生成器对象的send方法详解

    前言 在上一篇博客中,笔者带大家一起探讨了生成器与迭代器的本质原理和使用,本次博客将重点聚焦于生成器对象的send方法. 一.send方法详解  我们知道生成器对象本质上是一个迭代器.但是它比迭代器对 ...

  3. 可迭代对象&迭代器&生成器

    在python中,可迭代对象&迭代器&生成器的关系如下图: 即:生成器是一种特殊的迭代器,迭代器是一种特殊的可迭代对象. 可迭代对象 如上图,这里x是一个列表(可迭代对象),其实正如第 ...

  4. 【Python】【容器 | 迭代对象 | 迭代器 | 生成器 | 生成器表达式 | 协程 | 期物 | 任务】

    Python 的 asyncio 类似于 C++ 的 Boost.Asio. 所谓「异步 IO」,就是你发起一个 IO 操作,却不用等它结束,你可以继续做其他事情,当它结束时,你会得到通知. Asyn ...

  5. 深入理解python中可迭代对象,迭代器,生成器

    英文原文出处:Iterables vs. Iterators vs. Generators 在python学习中,通常会陷入对以下几个相关概念之间的确切差异的困惑中: a container(容器) ...

  6. Python--可迭代对象,迭代器,生成器

    记得在刚开始学Python的时候,看到可迭代对象(iterable).迭代器(iterator)和生成器(generator)这三个名词时,完全懵逼了,根本就不知道是啥意识.现在以自己的理解来详解下这 ...

  7. 闭包在python中的应用,translate和maketrans方法详解

    python对字符串的处理是比较高效的,方法很多.maketrans和translate两个方法被应用的很多,但是具体怎么用常常想不起来. 让我们先回顾下这两个方法吧: 1.s.translate(t ...

  8. Object对象的浅拷贝与深拷贝方法详解

    /* ===================== 直接看代码 ===================== */ <!DOCTYPE html> <html> <head& ...

  9. Python入门之获取当前所在目录的方法详解

    #本文给大家讲解的是使用python获取当前所在目录的方法以及相关示例,非常的清晰简单,有需要的小伙伴可以参考下 sys.path 模块搜索路径的字符串列表.由环境变量PYTHONPATH初始化得到. ...

随机推荐

  1. HTML5 原生拖放

    前言: HTML5提供专门的拖拽与拖放的API,可以方便的指定某个元素可拖动,可以创建自定义的可拖动元素和放置目标 相关知识点: 1.拖放事件 拖放元素时,将依次触发下列事件 dragstart  按 ...

  2. 品Spring:对@Resource注解的处理方法

    @Resource是Java的注解,表示一个资源,它具有双向的含义,一个是从外部获取一个资源,一个是向外部提供一个资源. 这其实就对应于Spring的注入和注册.当它用在字段和方法上时,表示前者.当它 ...

  3. 完美实现保存和加载easyui datagrid自定义调整列宽位置隐藏属性功能

    需求&场景 例表查询是业务系统中使用最多也是最基础功能,但也是调整最平凡,不同的用户对数据的要求也不一样,所以在系统正式使用后,做为开发恨不得坐在业务边上,根据他们的要求进行调整,需要调整最多 ...

  4. opencv之为图像添加边界

    我们经常会有对图像边缘做扩展的需求.比如 希望卷积后得到的矩阵大小不变 希望改变图像大小,但是不改变宽高比 opencv实现 opencv中使用copyMakeBorder()来完成这一功能 api ...

  5. 快学Scala 第二课 (apply, if表达式,循环,函数的带名参数,可变长参数,异常)

    apply方法是Scala中十分常见的方法,你可以把这种用法当做是()操作符的重载形式. 像以上这样伴生对象的apply方法是Scala中构建对象的常用手法,不再需要使用new. if 条件表达式的值 ...

  6. 快学Scala 第十四课 (读取行,读取字符, 控制台读取)

    读取行: import scala.io.Source object FileReader { def main(args: Array[String]): Unit = { val source = ...

  7. Docker5-docker私库的搭建及常用方法-harbor-registry方式

    一.简介 1.官方已经提供registry镜像为什么还需要用harbor 1)registry缺少镜像清理机制,可以push但是不能删除,耗费空间 2)registry缺乏相应的扩展机制 3)harb ...

  8. B/S 工业互联网 地铁行业

    前言 近几年,互联网与交通运输的融合,改变了交易模式,影响着运输组织和经营方式,改变了运输主体的市场结构.模糊了运营与非营运的界限,也更好的实现了交通资源的集约共享,同时使得更多依靠外力和企业推动交通 ...

  9. Java通信——获取自己IP

    获取自己的IP地址 import java.net.InetAddress; import java.net.UnknownHostException; public class getip { pu ...

  10. PHP 实现get 和 Post 请求

    1 get get请求比较简单,file_get_contents():即可实现 $tmpUrl = "http://测试url"; # get方法获取信息 $rawGetData ...