[python]Generators
generators(生成器)是python提供的一种机制,可以让函数一边循环一边计算,通常函数是一遍执行,而生成器可以在执行中间交出变量,下次调用时从交出变量的地方重新开始,这种机制通过yield关键字实现。
考虑生成斐波那契数,即1,1,2,3,5,8......、
假设我不指定生成的项数,而通过调用next()的方法不断循环下一个数,那么代码应该如下:
- class Fib(object):
- def __init__(self):
- self.a, self.b = 0, 1
- def __iter__(self):
- return self
- def next(self):
- self.a, self.b = self.b, self.a + self.b
- if self.a > 100000:
- raise StopIteration()
- return self.a
实际上这么编写的Fib类已经实现了迭代器(Iterator)的功能,因为实现了__iter__()和next()方法,在一个实例中调用next()方法可以不断的得到下一个斐波拉契数,而用for循环时则会先调用__iter__()方法,该方法返回一个实例,再去调用该实例的next()方法。
其实这里好像已经实现不断迭代的功能了(笑),然而我们一开始是要用生成器来实现的,那生成器应该怎么做。
- def fib():
- a,b=0,1
- while(a<100000):
- a,b=b,a+b
- yield a
如此就编写了一个生成器,代码真是精简(捂脸)......
同样的,a=fib()后,a可以调用next()方法,并且a也可以用for循环,结果与迭代器一致。
注意这里如果直接调用fib().next()则每次输出都是1,1,1,1,1,1
打印一下fib().next看看
fib().next
<method-wrapper 'next' of generator object at 0x0000000003AEFEE8>
fib().next
<method-wrapper 'next' of generator object at 0x0000000003B553F0>
可以看到两次的地址不一致,其实就是创建了一个方法而已,再看看fib跟fib()的区别:
fib
<function fib at 0x0000000003B4FAC8>
fib()
<generator object fib at 0x0000000003AEFD38>
一个是函数,一个是生成器。
因此a=fib是指向了fib方法,a=fib()才是”继承“了一个生成器。
在廖雪松的Python教程中生成器一节有一个杨辉三角练习
- def triangles():
- L=[1]
- while(True):
- yield L
- L.append(0)
- L = [L[i-1]+L[i] for i in range(len(L))]
- n=0
- for t in triangles():
- print(t)
- n = n + 1
- if n == 10:
- break
打印结果为:
- [1]
- [1, 1]
- [1, 2, 1]
- [1, 3, 3, 1]
- [1, 4, 6, 4, 1]
- [1, 5, 10, 10, 5, 1]
- [1, 6, 15, 20, 15, 6, 1]
- [1, 7, 21, 35, 35, 21, 7, 1]
- [1, 8, 28, 56, 70, 56, 28, 8, 1]
- [1, 9, 36, 84, 126, 126, 84, 36, 9, 1]
这段代码没有难度,不过里面用到的一个小技巧就是L[-1]+L[0]中L[-1]其实是L的最后一个元素,也就是新添加的0,这样十分巧妙地将需要往L中首尾添加0的情形变为只要末尾添加0就行了,十分的耐人寻味。
yield的用法其实不止体现在生成器中,在Python的上下文管理中也有体现。例如某个过程的上下文管理如下:
- from contextlib import contextmanager
- @contextmanager
- def context():
- print "Begin"
- yield
- print "End"
- with context():
- print "a process"
因为context()实现了上下文管理,所以可以用with来调用,当执行倒yield时,函数退出交给实际的process处理,完成后再继续调用yield后面的语句结束。输出结果为:
- Begin
- a process
- End
总结:
Python中生成器的编写主要是用到了yield关键字,让函数能够”停下来“一次一次地输出,迭代器能实现同样的功能,但迭代器要实现__iter__()和next()方法。
http://www.liaoxuefeng.com/wiki/0014316089557264a6b348958f449949df42a6d3a2e542c000/0014317799226173f45ce40636141b6abc8424e12b5fb27000
http://blog.csdn.net/scelong/article/details/6969276
[python]Generators的更多相关文章
- (转) Python Generators(生成器)——yield关键字
http://blog.csdn.net/scelong/article/details/6969276 生成器是这样一个函数,它记住上一次返回时在函数体中的位置.对生成器函数的第二次(或第 n 次) ...
- Python Generators(生成器)--yield
参考:http://blog.csdn.net/scelong/article/details/6969276 Python生成器 什么是python生成器,意思是带有一个yield语句的函数,既然它 ...
- Python Generators vs Iterators
http://stackoverflow.com/questions/2776829/difference-between-python-generators-vs-iterators iterato ...
- Python资源大全
The Python Tutorial (Python 2.7.11) 的中文翻译版本.Python Tutorial 为初学 Python 必备官方教程,本教程适用于 Python 2.7.X 系列 ...
- Python’s SQLAlchemy vs Other ORMs[转发 5] PonyORM
PonyORM PonyORM allows you to query the database using Python generators. These generators are trans ...
- 十二. Python基础(12)--生成器
十二. Python基础(12)--生成器 1 ● 可迭代对象(iterable) An object capable of returning its members one at a time. ...
- python coroutine的学习跟总结[转]
简介 因为最近一段时间需要研究一些openstack相关的东西,在阅读一些相关代码的时候碰到很多python特定的一些特性,比如generator, coroutine以及一些相关的类库,比如even ...
- Python 协程与事件循环
Table of Contents 前言 协程 async & await 事件循环 asyncio 的事件循环 结语 参考链接 前言 Python 标准库 asyncio 是我目前接触过的最 ...
- [Python] Use a Python Generator to Crawl the Star Wars API
In this lesson, you will be introduced to Python generators. You will see how a generator can replac ...
随机推荐
- day 17 项目开发常用模块
---恢复内容开始--- time模块 import time print(time.time()) # 时间戳: print(time.strftime("%Y-%m-%d %X" ...
- django 分页出现 UnorderedObjectListWarning 错误
django 分页出现此错误: UnorderedObjectListWarning: Pagination may yield inconsistent results with an unorde ...
- vue与mapbox
1.引入mapbox 在index.html里面引入maobox gl 或者使用 npm install mapbox-gl 2.功能 我主要这次要实现 地图展示 地图打点(添加Marker) 地图的 ...
- yum 的 group的信息
查找 yum源中有哪些group及其详细信息 1:yum groupinfo '*' | less 2:yum groupinfo '*' | less +/sendmail-cf 将 sendmai ...
- springboot中get post put delete 请求
组合注解(RequestMapping的变形) @GetMapping = @RequestMapping(method = RequestMethod.GET) @PostMapping = @Re ...
- #20175120彭宇辰 java第四周总结
教材学习内容总结 -第五章 子类与继承 子类的定义-class 子类名 extends 父类名ps:-任何类都是object的子类-每个类有且仅有一个父类,多个或0个子类 同一包里的继承性-除了pri ...
- 网络编程之OSI七层协议略析
OSI七层协议 1 物理连接层 用高低电平模拟二进制数01012 数据链路层 以太网协议,Ethernet,要求连入网络的计算机必要要有一块网卡,并且规定全球每一块网卡的mac 地址都不一样,用12位 ...
- ViewpageMaiActity
<?xml version="1.0" encoding="utf-8"?><RelativeLayout xmlns:android=&qu ...
- python: 文件的读写
#文件的读取.py a=open('test.txt').readline() #只读取文件第一行,保存为字符串格式 b=open('test.txt').read() #读取全部内容,保存为字符串格 ...
- 用Redis存储Tomcat集群的Session实现session共享
一.存储 前段时间,我花了不少时间来寻求一种方法,把新开发的代码推送到到生产系统中部署,生产系统要能够零宕机.对使用用户零影响. 我的设想是使用集群来搞定,通过通知负载均衡Nginx,取下集群中的To ...