原文发表在个人博客主页,转载请注明出处html
没有用过的东西,没有深入理解的东西很难说本身会,并且被别人一问必然破绽百出。虽然以前有接触过python协程的概念,可是只是蜻蜓点水,这两天的一次交谈中,别人问到了协程,顿时语塞,死活想不起来曾经看过的东西,以后忽然想到了yield,但为时已晚,只能说概念不清,因此本篇先缕缕python的生成器和yield关键字。python
python提供了两种基本的方式。数组
下面详细讲解。函数
为何叫生成器函数?由于他随着时间的推移生成了一个数值队列。通常的函数在执行完毕以后会返回一个值而后退出,可是生成器函数会自动挂起,而后从新拾起继续执行,他会利用yield关键字关起函数,给调用者返回一个值,同时保留了当前的足够多的状态,可使函数继续执行。生成器和迭代协议是密切相关的,可迭代的对象都有一个__next()__成员方法,这个方法要么返回迭代的下一项,要么引发异常结束迭代。
为了支持迭代协议,拥有yield语句的函数被编译为生成器,这类函数被调用时返回一个生成器对象,返回的对象支持迭代接口,即成员方法__next()__继续从中断处执行执行。
看下面的例子:spa
# codes def create_counter(n): print "create counter" while True: yield n print 'increment n' n += 1 cnt = create_counter(2) print cnt print next(cnt) print next(cnt) # output <generator object create_counter at 0x0000000001D141B0> create counter 2 increment n 3
分析一下这个例子:code
为了更加深入的理解,咱们再举一个例子。协程
#coding def cube(n): for i in range(n): yield i ** 3 for i in cube(5): print i #output 0 1 8 27 64
因此从理解函数的角度出发咱们能够将yield类比为return,可是功能确实彻底不一样,在for循环中,会自动遵循迭代规则,每次调用next()函数,因此上面的结果不难理解。htm
生成器表达式来自于迭代和列表解析的组合,关于列表解析的概念和用法能够参见我以前的博客,生成器表达式和列表解析相似,可是他使用尖括号而不是方括号括起来的。以下代码:对象
>>> # 列表解析生成列表 >>> [ x ** 3 for x in range(5)] [0, 1, 8, 27, 64] >>> >>> # 生成器表达式 >>> (x ** 3 for x in range(5)) <generator object <genexpr> at 0x000000000315F678> >>> # 二者之间转换 >>> list(x ** 3 for x in range(5)) [0, 1, 8, 27, 64]
就操做而言,生成器表若是使用大量的next()函数会显得十分不方便,for循环会自动出发next函数,因此能够按下面方式使用:blog
>>> for n in (x ** 3 for x in range(5)): print('%s, %s' % (n, n * n)) 0, 0 1, 1 8, 64 27, 729 64, 4096 >>>
一个迭代既能够被写成生成器函数,也能够被协程生成器表达式,均支持自动和手动迭代。并且这些生成器只支持一个active迭代,也就是说生成器的迭代器就是生成器自己。