14.Python生成器
生成器
引入
在Python中,使用生成器可以很方便的支持迭代器协议。生成器通过生成器函数产生,生成器函数可以通过常规的def语句来定义,但是不用return返回,而是用yield一次返回一个结果,在每个结果之间挂起和继续它们的状态,来自动实现迭代协议。
也就是说,yield是一个语法糖,内部实现支持了迭代器协议,同时yield内部是一个状态机,维护着挂起和继续的状态。
1 | def my_range(n): |
在这个例子中,定义了一个生成器函数,函数返回一个生成器对象,然后就可以通过for语句进行迭代访问了。
其实,生成器函数返回生成器的迭代器。 “生成器的迭代器”这个术语通常被称作”生成器”。要注意的是生成器就是一类特殊的迭代器。作为一个迭代器,生成器必须要定义一些方法,其中一个就是__next__()。如同迭代器一样,我们可以使用__next__()函数来获取下一个值。
生成器工作流程
下面就仔细看看生成器是怎么工作的。从上面的例子也可以看到,生成器函数跟普通的函数是有很大差别的。结合上面的例子我们加入一些打印信息,进一步看看生成器的执行流程:
1 | def my_range(n): |
通过结果可以看到:
当调用生成器函数的时候,函数只是返回了一个生成器对象,并没有执行。
当
next()方法第一次被调用的时候,生成器函数才开始执行,执行到yield语句处停止:next()方法的返回值就是yield语句处的参数yielded value当继续调用
next()方法的时候,函数将接着上一次停止的yield语句处继续执行,并到下一个yield处停止;如果后面没有yield就抛出StopIteration异常
生成器表达式
在开始介绍生成器表达式之前,先看看我们比较熟悉的列表解析[List comprehensions],列表解析一般都是下面的形式。
1 | [expr for iter_var in iterable if cond_expr] |
迭代iterable里所有内容,每一次迭代后,把iterable里满足cond_expr条件的内容放到iter_var中,再在表达式expr中获取iter_var的内容,最后用表达式的计算值生成一个列表。
例如,生成一个list包含50以内的所有奇数:
1 | [i for i in range(50) if i % 2] |
生成器表达式是在python2.4中引入的,当序列过长, 而每次只需要获取一个元素时,应当考虑使用生成器表达式而不是列表解析。生成器表达式的语法和列表解析一样,只不过生成器表达式是被()括起来的,而不是[],如下:
1 | (expr for iter_var in iterable if cond_expr) |
生成器表达式并不是创建一个列表, 而是返回一个生成器,这个生成器在每次计算出一个条目后,把这个条目”产生”(yield)出来。 生成器表达式使用了”惰性计算”(lazy evaluation),只有在检索时才被赋值(evaluated),所以在列表比较长的情况下使用生成器会在内存上更有效。
1 | gen = (i for i in range(10000) if i % 2) |
生成器中的send()与close()方法
生成器中还有两个很重要的方法:send()和close()。
send(value):从前面了解到,
next()方法可以恢复生成器状态并继续执行,其实send()是除next()外另一个恢复生成器的方法。Python 2.5中,yield语句变成了yield表达式,也就是说yield可以有一个值,而这个值就是send()方法的参数,所以send(None)和next()是等效的。同样,next()和send()的返回值都是yield语句处的参数(yielded value)关于
send()方法需要注意的是:调用send传入非None值前,生成器必须处于挂起状态,否则将抛出异常。也就是说,第一次调用时,要使用next()语句或send(None),因为没有yield语句来接收这个值。close():这个方法用于关闭生成器,对关闭的生成器后再次调用
next或send将抛出StopIteration异常。
1 | def my_range(n): |
总结:
- 生成器是一种特殊的迭代器,内部支持了生成器协议,不需要明确定义
__iter__()和__next__()方法。 - 生成器通过生成器函数产生,生成器函数可以通过常规的def语句来定义,但是不用
return返回,而是用yield一次返回一个结果。
- 标题: 14.Python生成器
- 作者: Jinshuo Jiang
- 创建于 : 2025-09-30 21:57:13
- 更新于 : 2025-09-30 21:58:10
- 链接: https://redefine.ohevan.com/2025/09/30/Python-Generator/
- 版权声明: 本文章采用 CC BY-NC-SA 4.0 进行许可。