迭代器iter

1.迭代的含义:

每次生成的结果依赖于上一次。问路,先问第一个人,第
一个人不知道他就说第二个人知道,然后去找第二个人。第二个人不知道
就说第三个人知道,然后去找第三个人

2.递归的含义:

问路,先问第一个人,不知道就第一个人问第二个人,还
不知道就第二个人问第三个,还不知道就第三个人问第四个。如果第四
个知道了就告诉第三个,第三个告诉第二个,一直反复
然后告诉给原来问路的那个人

3.迭代器协议:

对象必须要提供一个next的下一步的方法。就像生孩子传宗接代一样,要按着
一个方法一代接一代下去,也有可能会出现异常,终止迭代。(只能接着往下
走没有后退的路)

4.可迭代对象:实现了迭代器协议的对象。

5.for循环只能遍历可迭代对象。

(字符串,列表,元组,字典,集合,文件对象)这些其实不是可迭代对象,因为
他们没有.next的附加方法,但是在for调用他们的时候,调用了他们内部的__iter__
方法,把他们都变成了可迭代对象。

x='hello'      #变字符串的成可迭代
diedai = x.__iter__()
print(diedai.__next__()) #本句也可以改成print(next(diedai))
print(diedai.__next__())
dic={'a':1,'b':2} #变字典的成可迭代
died=dic.__iter__()
print(died.__next__())
print(died.__next__())
died1=dic.values().__iter__()
print(died1.__next__())
print(died1.__next__())

生成器

1.生成器定义:

生成器是一种数据类型,这种数据类型自动实现了迭代器协议
,不用执行iter方法,它本身就是可迭代对象。它可以直接用.next()。生成器很
节省内存,next()显示下一个之后上一个会被删掉。

2.生成器在python内有两种表达方式

第一种:

生成器函数,在函数内用yield语句而不是return可以将函数变成生成
器函数。

def test():
yield 1
yield 'sb'
yield 'nt'
g=test()
print(g.__next__())
print(g.__next__())

第二种:生成器表达式

首先需要了解三元表达式

#三元表达式,先看中间再看左边最后右边

name='alex'
res='sb' if name == 'alex' else 'shuai' #本句为3元表达式
#如果名字是alex就把sb赋值给res,if的结果写在前面,而else的结果帅写后面
print(res)

还需要了解一下列表解析

#列表解析
egglist=[] #建立一个空的列表
for i in range(10):
egglist.append('鸡蛋%s' %i)
print(egglist)
#接下来用列表解析优化
l=['鸡蛋%s' %i for i in range(10)]
#用列表解析和三元表达式弄出标签大于5的鸡蛋
l1=['鸡蛋%s' %i for i in range(10) if i>5]
print(l)
print(l1)

接下来来说生成器表达式

生成器表达式就是列表解析把[]改成()

laomuji=('鸡蛋%s'  %i for i in range(10))
#此时laomuji就已经是生成器表达式,可迭代对象
print(laomuji.__next__())
print(laomuji.__next__())
print(laomuji.__next__())
print(laomuji.__next__())

实例:现做现卖包子

#使用__next__()是走到yield的位置就跳出,下一次next从当前跳出位置开始
def product_baozi():
for i in range(1,100): #循环做包子
print('正在生产包子')
yield '第 %s 个包子' %i #第一次调用next来到这跳出,下一次next调用从这开始
print('开始卖包子')
pro_g=product_baozi()
print(pro_g.__next__())
print(pro_g.__next__())
print(pro_g.__next__())

实例2:统计出字典中的每个地方的人口占据的人口比例

人口.txt内的内容
{'name':'北京','population':'10'}
{'name':'山东','population':'1000'}
{'name':'山西','population':'30010'}
{'name':'河北','population':'15200'}
{'name':'台湾','population':'750'}

程序:

#采用readline函数进行实现 算出人口比例
f=open('人口','r',encoding='utf8')
ret1=[]
ret=[]
i=0
i1=0
while i<5:
ret=eval(f.readline())
ret1.append(int(ret['population']))
i=i+1
qiuhe=sum(ret1)
print(qiuhe)
while i1<5:
print('%s %%' %(ret1[i1]/qiuhe))
i1=i1+1

#采用生成器来做

def getpopulation():
f=open('人口','r',encoding='utf8')
for i in f:
yield i
g=getpopulation()
qiuhe=0
qiuhe=sum(int(eval(i1)['population']) for i1 in g) #先看右边生成器迭代取每列,然后
print(qiuhe) #看左边把每列转换字符串,取人口,变数字型
g=getpopulation()
for p in g:
pnumber=int(eval(p)['population'])
print(pnumber/qiuhe)

生产消费者模型:

两者是并发运行的,一个生产然后跳到另一个消费,如此循环
采用生成器yield可以达到这个效果

1.首先需要了解启动生成器的第三种方法

之前只学过__next__和next()两种方法,还有一种方法用send

# def test():
# print('开始了')
# yield 1
# print('第一次')
# yield 2
# t=test()
# t.send(None) #用none的话和t.__next__()没区别
#send的优势是可以将一个值赋值给yield的那一行位置的形参
def test1():
print('开始了')
shit=yield 1
print('第一次',shit)
fuck=yield 2
print('第2次')
print(fuck)
yield 3
t=test1()
t.send(None) #光标运行至yield1
res2=t.send('m870') #此时m870将传递给yield1然后传给shit,而且光标将运行到yield2
t.send('m8') #当前光标在yield2,传m8给yeild2给fuck,而且光标运行到yield3
print('\n')
print(res2) #而且可以通过res2来接收yield的ruturn值

2.接下来来看并发的生产消费模型

import time
def consumer(name):
print('我是%s,我准备开始吃包子' %name)
while 1:
time.sleep(1)
print('包子生产好了')
baozi=yield
print('我是%s,我开始吃%s包子' %(name,baozi))
def producer(name,eattime):
c1=consumer(name) #启动生成器
c1.__next__() #光标移动至yield
for i in range(eattime):
time.sleep(1)
c1.send('第%s个屎馅' %i) #将屎馅传给yield传给baozi,因为while一直循环,光标到下一次yeild
producer('沙比',3)

Py迭代和迭代器,生成器,生产者和消费者模型的更多相关文章

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

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

  2. 生产者和消费者模型producer and consumer(单线程下实现高并发)

    #1.生产者和消费者模型producer and consumer modelimport timedef producer(): ret = [] for i in range(2): time.s ...

  3. Java线程(学习整理)--4---一个简单的生产者、消费者模型

     1.简单的小例子: 下面这个例子主要观察的是: 一个对象的wait()和notify()使用情况! 当一个对象调用了wait(),那么当前掌握该对象锁标记的线程,就会让出CPU的使用权,转而进入该对 ...

  4. Python之生产者&、消费者模型

    多线程中的生产者和消费者模型: 生产者和消费者可以用多线程实现,它们通过Queue队列进行通信. import time,random import Queue,threading q = Queue ...

  5. 【java线程系列】java线程系列之线程间的交互wait()/notify()/notifyAll()及生产者与消费者模型

    关于线程,博主写过java线程详解基本上把java线程的基础知识都讲解到位了,但是那还远远不够,多线程的存在就是为了让多个线程去协作来完成某一具体任务,比如生产者与消费者模型,因此了解线程间的协作是非 ...

  6. Spring MVC 使用介绍(七)—— 注解式控制器(三):生产者与消费者模型

    一.MIME类型 MIME类型格式:type/subtype(;parameter)? type:主类型,任意的字符串,如text,如果是*号代表所有 subtype:子类型,任意的字符串,如html ...

  7. 守护进程,互斥锁,IPC,队列,生产者与消费者模型

    小知识点:在子进程中不能使用input输入! 一.守护进程 守护进程表示一个进程b 守护另一个进程a 当被守护的进程结束后,那么守护进程b也跟着结束了 应用场景:之所以开子进程,是为了帮助主进程完成某 ...

  8. python queue和生产者和消费者模型

    queue队列 当必须安全地在多个线程之间交换信息时,队列在线程编程中特别有用. class queue.Queue(maxsize=0) #先入先出 class queue.LifoQueue(ma ...

  9. python并发编程之守护进程、互斥锁以及生产者和消费者模型

    一.守护进程 主进程创建守护进程 守护进程其实就是'子进程' 一.守护进程内无法在开启子进程,否则会报错二.进程之间代码是相互独立的,主进程代码运行完毕,守护进程也会随机结束 守护进程简单实例: fr ...

  10. 人生苦短之我用Python篇(队列、生产者和消费者模型)

    队列: queue.Queue(maxsize=0) #先入先出 queue.LifoQueue(maxsize=0) #last in fisrt out  queue.PriorityQueue( ...

随机推荐

  1. IIS本地部署局域网可随时访问的项目

    原理 在本机的IIS下创建一个网站,文件目录直接指向Web项目文件夹 步骤 1.项目的启动项目为web 2.在iis中创建一个新的网站(Work_TK_EIS) 文件目录为web项目的目录(D:\Gi ...

  2. [EF] - 动态创建模型:System.Reflection.Emit + Code First

    动态创建Entity Framework模型并且创建数据库 使用System.Reflection.Emit+Code First model创建以下的一个实体类和DbContext并且创建数据库: ...

  3. (四)、vim的缓冲区、标签、窗口操作

    1.缓冲区的基本操作 a.文件与缓冲区的区别 vim file1   打开一个文件时,其实是从磁盘中读取文件到内存中,文件的内容会被加载到缓冲区中, 这个缓冲区在一个窗口上显示,所以他是一个已激活的缓 ...

  4. 2.Redis info命令详解

    命令 127.0.0.1:6379> info [server|clients|memory|stats|...] # Server redis_version:5.0.4 #redis版本 r ...

  5. struts文件上传拦截器分析

    struts有默认的文件拦截器,一般配置maximumSize就可以了. 知道原理,我们可以写一个类继承它,实现自己的配置上传文件大小的方式.   然后细究页面上传文件的时候,发现了一些问题. act ...

  6. ElasticSearch设置用户名密码访问

    版本号:7.3.1 1.需要在配置文件中开启x-pack验证, 修改config目录下面的elasticsearch.yml文件,在里面添加如下内容,并重启. xpack.security.enabl ...

  7. 学习Promise异步编程

    JavaScript引擎建立在单线程事件循环的概念上.单线程( Single-threaded )意味着同一时刻只能执行一段代码.所以引擎无须留意那些"可能"运行的代码.代码会被放 ...

  8. FileZilla使用教程

    FileZilla使用教程和问题汇总 正式开始之前我还是想闲聊几句,毕竟调试解决这些问题,整整耗了我一天半的时间.前不久学校要求上交一个web大作业,然后我们宿舍就寻思光做出来前后端搞头不大,不如整个 ...

  9. std::thread线程详解(1)

    目录 目录 简介 线程的使用 线程的创建 线程的方法和属性 std::jthread (C++20) stop_token (C++20) 总结 Ref 简介 本文主要介绍了标准库中的线程部分.线程是 ...

  10. vue-element Form表单验证(表单验证没错却一直提示错误)

    在使用element-UI 的表单时,发生一个验证错误,例如已输入值但求验证纠错:       代码如下所示: <el-form :model="correction" :i ...