迭代器官方文档:https://docs.python.org/3/tutorial/classes.html#generators
大多是的文件对象可以用for循环语句,例如:
for element in [1, 2, 3]:
print(element)
for element in (1, 2, 3):
print(element)
for key in {'one':1, 'two':2}:
print(key)
for char in "123":
print(char)
for line in open("myfile.txt"):
print(line, end='')
这种风格是清晰简洁方便的。在其背后是for语句不断的调用容器对象的iter()方法,该方法返回一个迭代器对象,该对象定义了__next()__方法,该方法一次访问容器元素一次。当没元素访问时候,将抛出StopIteration的错误。我们可以通过调用next()的内置方法嗲用__next__()方法。
>>> s = 'abc'
>>> it = iter(s)
>>> it
<iterator object at 0x00A1DB50>
>>> next(it)
'a'
>>> next(it)
'b'
>>> next(it)
'c'
>>> next(it)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
next(it)
StopIteration
知道原理之后很容易将迭代器的行为添加到自定义类中,定义一个__iter__()方法返回一个有__next()__方法的对象,例如:
class Reverse:
"""Iterator for looping over a sequence backwards."""
def __init__(self, data):
self.data = data
self.index = len(data)
def __iter__(self):
return self
def __next__(self):
if self.index == 0:
raise StopIteration
self.index = self.index - 1
return self.data[self.index]
>>> rev = Reverse('spam')
>>> iter(rev)
<__main__.Reverse object at 0x00A1DB50>
>>> for char in rev:
... print(char)
...
m
a
p
s
生成器官方文档:https://docs.python.org/3/tutorial/classes.html#generators
生成器是一个简单且强大的工具可以用来创造迭代器。他们看起来很像一般的函数,但是用yield的语句返回数据。每次next()哈数被调用,生成器就重新从上次执行处继续执行。
>>> for char in reverse('golf'):
... print(char)
...
f
l
o
g
def reverse(data):
for index in range(len(data)-1, -1, -1):
yield data[index]
任何生成器能做的时期,如前描述的基于类的迭代器也可以做到。自动创建的__iter__() 和__next__()方法是的生成器如此的简洁。