何为使用yield
Python 中的关键字?
比如说,我在试着理解这个代码1:
def _get_child_candidates(self, distance, min_dist, max_dist):
if self._leftchild and distance - max_dist < self._median:
yield self._leftchild
if self._rightchild and distance + max_dist >= self._median:
yield self._rightchild
这就是打电话的人:
result, candidates = [], [self]
while candidates:
node = candidates.pop()
distance = node._get_dist(obj)
if distance <= max_dist and distance >= min_dist:
result.extend(node._values)
candidates.extend(node._get_child_candidates(distance, min_dist, max_dist))
return result
当方法_get_child_candidates
是否调用 ? 列表是否返回 ? 单元素 ? 是否又调用 ? 以后的呼叫何时停止 ?
1. 本代码由Jochen Schulz(jrschulz)编写,他为公制空间制作了一个伟大的Python图书馆。模块 m 空间.
所有的答案都是伟大的, 但对于新人来说有点困难。
我猜你已经学会了return
语句。
作为类比,return
和yield
双胞胎。return
意指“返回和停止”,而“真正”意指“返回,但继续”
- 尝试获得 num_ list 列表
return
.
def num_list(n):
for i in range(n):
return i
运行它:
In [5]: num_list(3)
Out[5]: 0
你看,你只得到一个数字 而不是他们的名单。return
永远不允许你快乐地胜利, 仅仅一次执行,然后退出。
- 来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来来
yield
替换return
与yield
:
In [10]: def num_list(n):
...: for i in range(n):
...: yield i
...:
In [11]: num_list(3)
Out[11]: <generator object num_list at 0x10327c990>
In [12]: list(num_list(3))
Out[12]: [0, 1, 2]
现在,你赢得了所有的数字。
与return
运行一次,停止一次,yield
计划运行时间。您可以解释return
计为return one of them
, 和yield
计为return all of them
。这被称为iterable
.
- 再来一步,我们可以改写
yield
声明的语中return
In [15]: def num_list(n):
...: result = []
...: for i in range(n):
...: result.append(i)
...: return result
In [16]: num_list(3)
Out[16]: [0, 1, 2]
这是核心yield
.
列表之间的差别return
输出和对象yield
输出为 :
您将总是从列表对象中获取 [0, 1, 2] 列表对象, 但只能从“ 对象” 中获取它们yield
输出一次。 所以, 它有一个新名称generator
对象显示于Out[11]: <generator object num_list at 0x10327c990>
.
最后,作为格罗克语的比喻:
return
和yield
双胞胎
list
和generator
双胞胎
yield
简直就像return
区别在于,下次你打电话给发电机时,从最后一次呼叫开始执行。yield
与返回不同的语句,当生成时, 堆叠框架不会被清理, 但是控件会被转回调用方, 所以下次调用函数时, 它的状态将会恢复 。
对于您的代码,函数get_child_candidates
动作就像一个循环器,这样当您扩展列表时,它会一次向新列表添加一个元素。
list.extend
在你公布的代码样本中, 只需将图普还给列表, 并附加到列表中, 就会更加清晰 。
又一个TRL;DR
列表中的迭代器: next()
返回列表的下一个元素
热机发电机: next()
将计算苍蝇上的下一个元素( 执行代码)
您可以看到生成/生成器作为手动运行控制流量从外部( 如继续循环一步骤) 调用next
无论流量如何复杂。
Note发电机是不无一个普通函数。它会像本地变量( stack) 一样记得以前的状态( stack) 。请参看其他答案或文章以详细解释。生成器只能是曾经变热过一次. 你可以没有yield
,但它不会是那么好, 所以它可以被认为是“非常好”的语言糖。
通常情况下, 它会用来创建一个不起作用的代名词。 将“ ield” 当作您函数的附加件, 以及您作为数组的函数。 如果符合某些标准, 您可以在函数中添加此值, 使之成为代名词 。
arr=[]
if 2>0:
arr.append(2)
def func():
if 2>0:
yield 2
两者的输出结果相同。
使用产量的主要优势是创建迭代器。 迭代器在即时计算时不会计算每个项目的价值。 它们只在您要求时才计算。 这被称为懒惰评价 。