1.迭代器与生成器
迭代器(iterator)与生成器(generator)是 Python 中比较常用又很容易混淆的两个概念,今天就把它们梳理一遍,并举一些常用的例子。
2迭代器
for 语句与可迭代对象(iterable object):for循环就是基于迭代器协议提供了一个统一的可以遍历所有对象的方法,即在遍历之前,先调用对象的__iter__方法将其转换成一个迭代器,然后使用迭代器协议去实现循环访问,这样所有的对象就都可以通过for循环来遍历了,
for i in [123]:b
print(i)
输出:1
2
3
obj={'a':123,'b':456}
for i in obj:
print(i)
输出:a
b
这些可以用在 for
语句进行循环的对象就是可迭代对象。除了内置的数据类型(列表、元组、字符串、字典等)可以通过 for
语句进行迭代,我们也可以自己创建一个容器,包含一系列元素,可以通过 for
语句依次循环取出每一个元素,这种容器就是迭代器(iterator)。除了用 for
遍历,迭代器还可以通过 next()
方法逐一读取下一个元素。要创建一个迭代器有3种方法,其中前两种分别是:
1为容器对象添加 __iter__()
和 __next__()
方法(Python 2.7 中是 next()
);__iter__()
返回迭代器对象本身 self
,__next__()
则返回每次调用 next()
或迭代时的元素;
- 2内置函数
iter()
将可迭代对象转化为迭代器
-
# iter(IterableObject)ita = iter([1, 2, 3])print(type(ita))print(next(ita))print(next(ita))print(next(ita))# Create iterator Objectclass Container:def __init__(self, start = 0, end = 0):self.start = startself.end = enddef __iter__(self):print("[LOG] I made this iterator!")return selfdef __next__(self):print("[LOG] Calling __next__ method!")if self.start < self.end:i = self.startself.start += 1return ielse:raise StopIteration()c = Container(0, 5)for i in c:print(i)[LOG] I made this iterator![LOG] Calling __next__ method!0[LOG] Calling __next__ method!1[LOG] Calling __next__ method!2[LOG] Calling __next__ method!3[LOG] Calling __next__ method!4[LOG] Calling __next__ method!创建迭代器对象的好处是当序列长度很大时,可以减少内存消耗,因为每次只需要记录一个值即刻(经常看到人们介绍 Python 2.7 的
range
函数时,建议当长度太大时用xrange
更快,在 Python 3.5 中已经去除了xrange
只有一个类似迭代器一样的range
)。3生成器前面说到创建迭代器有3种方法,其中第三种就是生成器(generator)。生成器通过yield
语句快速生成迭代器,省略了复杂的__iter__()
&__next__()
方式:def container(start, end):while start < end:yield startstart += 1c = container(0, 5)print(type(c))print(next(c))next(c)for i in c:print(i)12345<class 'generator'>0234
简单来说,yield
语句可以让普通函数变成一个生成器,并且相应的 __next__()
方法返回的是 yield
后面的值。一种更直观的解释是:程序执行到 yield
会返回值并暂停,再次调用 next()
时会从上次暂停的地方继续开始执行:
def gen():
yield 5
yield "Hello"
yield "World"
yield 4
for i in gen():
print(i)
5
Hello
World
4