概述
def func():
yield 1**2
print('Hi, Im A!')
yield 2**2
print('Hi, Im B!')
yield 3**2
print('Hi, Im C!')
然后我们调用这个小函数,来看看yield产生的实际效果是什么:
>>> f = func()
>>> f
<generator object func at 0x10d36c840>
>>> next( f )
1
>>> next( f )
Hi, Im A!
4
>>> next( f )
Hi, Im B!
9
>>> next( f )
Hi, Im C!
ERROR: StopIteration
从这里我们可以看到:
第一次调用生成器的时候,yield之后的打印没有执行。因为程序yield这里暂停了
第二次调用生成器的时候,第一个yield之后的语句执行了,并且再次暂停在第二个yield
第三次调用生成器的时候,卡在了第三个yield。
第四次调用生成器的时候,最后一个yield以下的内容还是执行了,但是因为没有找到第四个yield,所以报错。
所以到了这里,如果我们能理解yield作为暂停符的作用,就可以非常灵活的用起来了。
yield from与sub-generator子生成器
yield from是Python 3.3开始引入的新特性。
它主要作用就是:当我需要在一个生成器函数中使用另一个生成器时,可以用yield from来简化语句。
举例,正常情况下我们可能有这么两个生成器,第二个调用第一个:
def gen1():
yield 11
yield 22
yield 33
def gen2():
for g in gen1():
yield g
yield 44
yield 55
yield 66
可以看到,我们在gen2()这个生成器中调用了gen1()的结果,并把每次获取到的结果yield转发出去,当成自己的yield出来的值。
我们把这种一个生成器中调用的另一个生成器叫做sub-generator子生成器,而这个子生成器由yield from关键字生成。
由于sub-generator子生成器很常用,所以Python引入了新的语法来简化这个代码:yield from。
上面gen2()的代码可以简化为:
yield from gen1()
yield 44
yield 55
yield 66
这样看起来是不是更"pythonic"了呢?:)
所以只要记住:yield from只是把别人呕吐出来的值,直接当成自己的值呕吐出去。
递归+yield能产生什么?
一般我们只是二选一:要不然递归,要不然for循环中yield。有时候yield就可以解决递归的问题,但是有时候光用yield并不能解决,还是要用递归。
那么怎么既用到递归,又用到yield生成器呢?
def func(n):
result = n**2
yield result
if n < 100:
yield from func( result )
for x in func(100):
print( x )
上面代码的逻辑是:如果n小于100,那么每次调用next(..)的时候,都得到n的乘方。下次next,会继续对之前的结果进行乘方,直到结果超过100为止。
我们看到代码里利用了yield from子生成器。因为yield出的值不是直接由变量来,而是由“另一个”函数得来了。
最后
以上就是凶狠小熊猫为你收集整理的yield generate python_Python高级语法之:一篇文章了解yield与Generator生成器的全部内容,希望文章能够帮你解决yield generate python_Python高级语法之:一篇文章了解yield与Generator生成器所遇到的程序开发问题。
如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。
发表评论 取消回复