Python3.5迭代器与生成器用法实例分析


Posted in Python onApril 30, 2019

本文实例讲述了Python3.5迭代器与生成器用法。分享给大家供大家参考,具体如下:

1、列表生成式

通过列表生成式可以直接创建一个列表。代码:a = [i*2 for i in range(10)]

#!/usr/bin/env python
# -*- coding:utf-8 -*-
# Author:ZhengzhengLiu
#列表生成式
a = [i*2 for i in range(10)]
print(a)

运行结果:

[0, 2, 4, 6, 8, 10, 12, 14, 16, 18]

由于受内存限制,列表容量肯定是有限的。创建一个包含100万个元素的列表,不仅占用很大的存储空间,若只访问前面的几个元素,后边的绝大多数元素占用空间浪费。

如果列表元素可以按照某种算法推算出来,那是否可以在循环过程中不断推算后续的元素?这样就不必创建完整的列表list,从而节省大量的空间。

2、生成器

在Python中,一边循环一边计算的机制,叫做:生成器(generator)。创建一个生成器的方法有很多:

(1)将一个列表生成式的[]改成(),就创建一个生成器。代码:b = (i*2 for i in range(10))

#!/usr/bin/env python
# -*- coding:utf-8 -*-
# Author:ZhengzhengLiu
#列表生成式
a = [i*2 for i in range(10)]
print(a)
print("type of a:",type(a))
#生成器
b = (i*2 for i in range(10))
print(b)
print("type of b:",type(b))
for i in b:
  print(i)

运行结果:

[0, 2, 4, 6, 8, 10, 12, 14, 16, 18]
type of a: <class 'list'>
<generator object <genexpr> at 0x008B8D20>
type of b: <class 'generator'>
0
2
4
6
8
10
12
14
16
18

结论:生成器的元素只有在调用的时候才生成相应的,调用到哪一次才会生成到哪一次的元素,只记住当前的位置。

注意:列表可以直接打印出每一个元素,而生成器不能用切片的形式去取,会出错误。

打印出生成器generator的每一个元素的方法:如果要一个一个打印出来,要通过next()函数获得生成器generator的下一个返回值

生成器generator保存的是算法,每次调用print(next(b)),就计算出生成器b的下一个元素的值,直到最后一个元素,没有更多的元素时,抛出StopIteration的错误。

#生成器
b = (i*2 for i in range(10))
print(next(b))
print(next(b))
print(next(b))
print(next(b))

运行结果:

0
2
4
6

不断调用next(b)很麻烦,可以利用for循环,因为生成器generator也是可迭代的对象

(2)当推算的算法比较复杂时,用类似列表生成式的for循环无法实现,还可以用函数来实现生成器

例如:著名的斐波那契数列(Fibonaccl),除了第一个和第二个数之外,任意一个数都由前两个数相加得到:1, 1, 2, 3, 5, 8, 13, 21, 34, ...

#!/usr/bin/env python
# -*- coding:utf-8 -*-
# Author:ZhengzhengLiu
def fibonaccl(max):
  n,a,b = 0,0,1
  while n < max:
    print(b)
    a,b = b,a + b
    n = n + 1
  return 'done'
fibonaccl(10)

运行结果:

1
1
2
3
5
8
13
21
34
55

Python3.5迭代器与生成器用法实例分析

总结:Fibonaccl函数实际上定义了斐波那契数列的推算规则,可以从第一个元素开始,推算出后续任意元素,这种逻辑非常类似generator。

Fibonaccl函数和生成器generator只有一步之遥,要把Fibonaccl函数变成生成器generator,只需要将print(b)修改为yield b就可以了。

最难理解的就是generator和函数的执行流程不一样。函数是顺序执行,遇到return语句或者最后一行函数语句就返回。

而变成generator的函数,在每次调用next()的时候执行,遇到yield语句返回,再次执行时从上次返回的yield语句处继续执行

即:yield保存了函数的中断状态,返回当前状态的值,函数停在这里,后边还可以继续回来。

另外,函数可以不再等待其执行结束,可以中断在某个地方做其他的事情,结束之后还可以继续回来接着往下执行(具有并行的效果)。

def fibonaccl(max):
  n,a,b = 0,0,1
  while n < max:
    yield b
    a,b = b,a + b
    n = n + 1
  return 'done'
print(fibonaccl(15))
f = fibonaccl(15)
print(f.__next__())
print(f.__next__())
print(f.__next__())
print(f.__next__())
print("===========")
print(f.__next__())
print(f.__next__())
print(f.__next__())
print(f.__next__())
print("=========start loop========")  #接着打印后边的元素
for i in f:
  print(i)

运行结果:

<generator object fibonaccl at 0x00548D50>
1
1
2
3
===========
5
8
13
21
=========start loop========
34
55
89
144
233
377
610

用for循环调用generator时,发现拿不到generator的return语句的返回值。

如果想要拿到返回值,必须捕获StopIteration错误,返回值包含在StopIteration的value中。

def fibonaccl(max):
  n,a,b = 0,0,1
  while n < max:
    yield b
    a,b = b,a + b
    n = n + 1
  return 'done'
g = fibonaccl(6)
while True:
   try:
     x = next(g)
     print('g:', x)
   except StopIteration as e:
     print('Generator return value:', e.value)
     break

运行结果:

g: 1
g: 1
g: 2
g: 3
g: 5
g: 8
Generator return value: done

3、生成器并行的实现——单线程下的并行效果

#!/usr/bin/env python
# -*- coding:utf-8 -*-
# Author:ZhengzhengLiu
#生成器并行的实现——生产者、消费者模型
import time
def consumer(name):
  print("%s 准备吃包子啦!" %name)
  while True:
    baozi = yield     #yield保存当前状态返回
    print("包子[%s]来了,被[%s]吃了!" %(baozi,name))
def producer(name):
  c = consumer('A')
  c2 = consumer('B')
  c.__next__()     #next只唤醒yield
  c2.__next__()
  print("开始准备做包子啦!")
  for i in range(3):
    time.sleep(1)
    print("做了2个包子!")
    c.send(i)     #send唤醒yield同时给它传值
    c2.send(i)
producer("alex")

运行结果:

A 准备吃包子啦!
B 准备吃包子啦!
开始准备做包子啦!
做了2个包子!
包子[0]来了,被[A]吃了!
包子[0]来了,被[B]吃了!
做了2个包子!
包子[1]来了,被[A]吃了!
包子[1]来了,被[B]吃了!
做了2个包子!
包子[2]来了,被[A]吃了!
包子[2]来了,被[B]吃了!

更多关于Python相关内容可查看本站专题:《Python数据结构与算法教程》、《Python Socket编程技巧总结》、《Python函数使用技巧总结》、《Python字符串操作技巧汇总》及《Python入门与进阶经典教程》

希望本文所述对大家Python程序设计有所帮助。

Python 相关文章推荐
python编写的最短路径算法
Mar 25 Python
使用beaker让Facebook的Bottle框架支持session功能
Apr 23 Python
Windows下Python使用Pandas模块操作Excel文件的教程
May 31 Python
python+pyqt实现右下角弹出框
Oct 26 Python
Python爬虫设置代理IP的方法(爬虫技巧)
Mar 04 Python
python3.x实现发送邮件功能
May 22 Python
基于Pandas读取csv文件Error的总结
Jun 15 Python
Python如何筛选序列中的元素的方法实现
Jul 15 Python
python中with语句结合上下文管理器操作详解
Dec 19 Python
Flask 上传自定义头像的实例详解
Jan 09 Python
Python如何在bool函数中取值
Sep 21 Python
Python获取指定日期是"星期几"的6种方法
Mar 13 Python
python使用Paramiko模块实现远程文件拷贝
Apr 30 #Python
Django学习笔记之为Model添加Action
Apr 30 #Python
Python Opencv实现图像轮廓识别功能
Mar 23 #Python
python opencv实现图像边缘检测
Apr 29 #Python
Python Django给admin添加Action的方法实例详解
Apr 29 #Python
python实现微信机器人: 登录微信、消息接收、自动回复功能
Apr 29 #Python
Python opencv实现人眼/人脸识别以及实时打码处理
Apr 29 #Python
You might like
风格模板初级不完全修改教程
2006/10/09 PHP
探讨PHP调用时间格式的参数详解
2013/06/06 PHP
php中HTTP_REFERER函数用法实例
2014/11/21 PHP
phpstudy隐藏index.php的方法
2020/09/21 PHP
firefox下对ajax的onreadystatechange的支持情况分析
2009/12/14 Javascript
在UpdatePanel内jquery easyui效果失效的解决方法
2010/04/11 Javascript
De facto standard 世界上不可思议的事实标准
2010/08/29 Javascript
Jquery拖拽并简单保存的实现代码
2010/11/28 Javascript
你必须知道的Javascript知识点之&quot;this指针&quot;的应用
2013/04/23 Javascript
jquery 追加tr和删除tr示例代码
2013/09/12 Javascript
js中string转int把String类型转化成int类型
2014/08/13 Javascript
node.js中的fs.fchown方法使用说明
2014/12/16 Javascript
jQuery基础_入门必看知识点
2016/07/04 Javascript
JavaScript随机打乱数组顺序之随机洗牌算法
2016/08/02 Javascript
JavaScript中英文字符长度统计方法示例【按照中文占2个字符】
2017/01/17 Javascript
angular实现IM聊天图片发送实例
2017/05/08 Javascript
详解使用angular的HttpClient搭配rxjs
2017/09/01 Javascript
Vue传参一箩筐(页面、组件)
2019/04/04 Javascript
JavaScript 面向对象基础简单示例
2019/10/02 Javascript
JavaScript 链表定义与使用方法示例
2020/04/28 Javascript
Vue 解决通过this.$refs来获取DOM或者组件报错问题
2020/07/28 Javascript
在react项目中使用antd的form组件,动态设置input框的值
2020/10/24 Javascript
Python使用pymysql小技巧
2017/06/04 Python
Python图片裁剪实例代码(如头像裁剪)
2017/06/21 Python
TensorFlow实现卷积神经网络CNN
2018/03/09 Python
python2.6.6如何升级到python2.7.14
2018/04/08 Python
Python lambda表达式filter、map、reduce函数用法解析
2019/09/11 Python
在tensorflow中设置保存checkpoint的最大数量实例
2020/01/21 Python
Python unittest单元测试框架及断言方法
2020/04/15 Python
GAP美国官网:美国休闲时尚品牌
2016/08/26 全球购物
印度服装购物网站:Limeroad
2018/09/26 全球购物
教师现实表现材料
2014/02/14 职场文书
小学社会实践活动总结
2014/07/03 职场文书
原生JavaScript实现简单五子棋游戏
2021/06/28 Javascript
MySQL系列之十二 备份与恢复
2021/07/02 MySQL
MySQL提升大量数据查询效率的优化神器
2022/07/07 MySQL