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的更多相关文章

  1. (转) Python Generators(生成器)——yield关键字

    http://blog.csdn.net/scelong/article/details/6969276 生成器是这样一个函数,它记住上一次返回时在函数体中的位置.对生成器函数的第二次(或第 n 次) ...

  2. Python Generators(生成器)--yield

    参考:http://blog.csdn.net/scelong/article/details/6969276 Python生成器 什么是python生成器,意思是带有一个yield语句的函数,既然它 ...

  3. Python Generators vs Iterators

    http://stackoverflow.com/questions/2776829/difference-between-python-generators-vs-iterators iterato ...

  4. Python资源大全

    The Python Tutorial (Python 2.7.11) 的中文翻译版本.Python Tutorial 为初学 Python 必备官方教程,本教程适用于 Python 2.7.X 系列 ...

  5. Python’s SQLAlchemy vs Other ORMs[转发 5] PonyORM

    PonyORM PonyORM allows you to query the database using Python generators. These generators are trans ...

  6. 十二. Python基础(12)--生成器

    十二. Python基础(12)--生成器 1 ● 可迭代对象(iterable) An object capable of returning its members one at a time. ...

  7. python coroutine的学习跟总结[转]

    简介 因为最近一段时间需要研究一些openstack相关的东西,在阅读一些相关代码的时候碰到很多python特定的一些特性,比如generator, coroutine以及一些相关的类库,比如even ...

  8. Python 协程与事件循环

    Table of Contents 前言 协程 async & await 事件循环 asyncio 的事件循环 结语 参考链接 前言 Python 标准库 asyncio 是我目前接触过的最 ...

  9. [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 ...

随机推荐

  1. Spring 源码学习(4)—— bean的加载part 1

    前面随笔中,结束了对配置文件的解析工作,以及将配置文件转换成对应的BeanDefinition存储在容器中.接下来就该进行bean的加载了. public Object getBean(String ...

  2. @Configuration的使用

    以下内容转载自:duanxz的spring4.0之二:@Configuration的使用,如有侵权,请联系作者本人予以删除 从Spring3.0,@Configuration用于定义配置类,可替换xm ...

  3. TestLink工具使用手册介绍

    工具名称:TestLink 工具介绍:TestLink遵循Apache2开源协议,免费试用.TestLink用于进行测试过程中的管理,通过使用TestLink提供的功能,可以将测试过程从测试需求.试设 ...

  4. shell中括号的特殊用法 linux if多条件判断

    一.   bash [  ] 单双括号 基本要素: Ø  [ ] 两个符号左右都要有空格分隔 Ø  内部操作符与操作变量之间要有空格:如  [  “a”  =  “b”  ] Ø  字符串比较中,&g ...

  5. webpack初步学习

    https://segmentfault.com/a/1190000006178770 该篇文章足够webpack入门的学习了,对webpack有个初步的了解和认识.

  6. Golang基础之函数

    golang基础之函数 1.为什么需要函数? 有些相同的代码可能出现多次,如果不进行封装,那么多次写入到程序中,会造成程序冗余,并且可读性降低 2.什么是函数 为完成某些特定功能的程序指令集合称为函数 ...

  7. 解决使用C/C++配置ODBC链接中文显示为问号(?)的问题

    使用VS2015中使用OBDC连接到数据库时,数据库可以正常显示,但是在VS上输出是乱码,如图: 在数据库中course表显示: vs程序结果显示: 查找原因,因为char默认读ascii型,只读到1 ...

  8. 与C/C++关键字extern有关的原理

    关键字有一定的语义,但是用法不唯一. 对于C/C++语言的预编译.编译.汇编.链接.我相信大家在接触C++一年不到就背的滚瓜烂熟,但是其中的细节,是后来才慢慢想明白的.为什么我不讲extern关键字呢 ...

  9. SpringAOP 注解方式

    Spring-service-mvc.xml <context:component-scan base-package="com.restful.controller,com.rest ...

  10. 【SpringBoot】Logback日志框架介绍和SpringBoot整合实战

    ========================11.Logback日志框架介绍和SpringBoot整合实战 2节课================================ 1.新日志框架L ...