python生成器

基础知识

生成器是python的一个特别特的特性,在许多场合都有重要应用。比如range函数产生的就是一个生成器。其主要的好处就是降低了内存的占用。为什么呢?拿range函数来讲吧,它的目的是生成一系列的数。假如我们想生成一列数0,1,2,3,4,则(以下两个紧跟的代码块中,第一个是实际的代码,第二个是输出)

range(5)
range(0, 5)

欸?怎么回事,输出的不是0,1,2,3,4,而是一个函数。这个函数实际上便是一个生成器。我们可以采用下面的方法把它展开

list(range(5))
[0, 1, 2, 3, 4]

或者使用一个for循环

for i in range(5):
    print(i)
0
1
2
3
4

生成器常常与for循环结合使用。在上述循环过程中,每循环一步,range产生一个数,直至range抛出一个结束的异常,for捕获异常后结束循环。
通过运行步骤,我们可以明白生成器节省空间的道理所在。range并没有一次性将数列都生成,而是逐渐生成。

自制生成器

为了进一步掌握生成器的编写方式,我们一起来实现range函数。当然这不一定是python内部的实现。如前文所述,range函数的作用是生成一列数。range的原本功能可以指定起始数、终止数和间隔。为了演示突出生成器的重点,我们将实现一个简化版的range,只给一个输入n,让其生成由0到n-1的数。并且,这个函数只是我们自己使用,就不做输入合法性检查了。闲言少叙,先看代码

def _range(n):
    step = 0
    while step < n:
        yield step
        step = step + 1
print(list(_range(5)))
[0, 1, 2, 3, 4]

注释讲解版函数如下

# 为了与python内置range函数进行区分,我们将其命名为_range.
# 那么如果你说我就不区分会怎么样?如果不区分的话,那么咱在这里定义的函数便会覆盖原函数。
def _range(n):
    # 采用一个变量暂存步数
    step = 0
    # 采用一个while循环,产生数据。
    # 当步数小于输入值n时,执行循环迭代
    while step < n:
        # 这是迭代器最关键的一步。yield表示输出。
        # 每当调用一次_range,便会执行到一个yield处,产生一个数据,同时暂停函数运行,直至再次调用。
        # 稍后会给出另外一个版本,加强大家在这方面的理解。
        yield step
        # 当再次调用时,会从这里开始,而非函数头部。步数加1,继续迭代
        step = step + 1
# 调用并将产生的数据展开后输出
print(list(_range(5)))

为了进一步加强大家对生成器的理解,接下来我们将上述版本稍作修改。我们想要产生一个0,0,1,1,2,2.。。。n-1,n-1的数列。该如何是好?且看

def _range(n):
    step = 0
    while step < n:
        yield step
        yield step
        step = step + 1
print(list(_range(5)))
[0, 0, 1, 1, 2, 2, 3, 3, 4, 4]

同时采用了两个yield,每当对_range函数调用一次,便从函数暂停的位置运行至下一个yield处,产生数据,随后输出。
接下来再举一个例子。我们产生一个0,0,1,0,1,2,0,1,2,3等等。这就要用到yield from这个语法了。它是生成器中的生成器。

def _range(n):
    step = 0
    while step < n:
        # range是另外一个生成器,yield from表示函数运行到此处时,进入range函数继续执行,直至range函数执行完毕。
        yield from range(step)
        step = step + 1
print(list(_range(5)))
[0, 0, 1, 0, 1, 2, 0, 1, 2, 3]

如果不用yield from会怎么样?想想。答案如下:

def _range(n):
    step = 0
    while step < n:
        # range是另外一个生成器,yield from表示函数运行到此处时,进入range函数继续执行,直至range函数执行完毕。
        yield range(step)
        step = step + 1
print(list(_range(5)))
for i in _range(5):
    print(i)
for i in _range(5):
    print(list(i))
[range(0, 0), range(0, 1), range(0, 2), range(0, 3), range(0, 4)]
range(0, 0)
range(0, 1)
range(0, 2)
range(0, 3)
range(0, 4)
[]
[0]
[0, 1]
[0, 1, 2]
[0, 1, 2, 3]

一个实际应用

不知道大家对生成器的基本原理及用法是否理解?下面我用一个更为实际的例子讲解生成器的妙用。
我在做数据处理时,单个数据文件可能非常大,直接读进内存很困难。这时候,便可采用生成器逐步读取处理。如

import os
def getFrame(filename):
    pos = 0
    frameSize = 100
    fullSize = os.path.getsize(filename)
    with open(filename, 'rb') as fid:
        while pos < fullSize:
            fid.seek(pos)
            yield fid.read(frameSize)
            pos = pos + frameSize

for frame in getFrame('SomeFilePath'):
    print(frame)

对于以上代码,我不过多讲解。大家对于其中有不明白的地方,可以自行查找帮助手册或者百度。如果还有不明白的地方,可以留言。其中大家可以重点理解with用法,这是在文件读取时常用并且非常好用的一个语法。
jupyter notebook原文件位于https://gitee.com/bolang/python-lesson.git

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,753评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,668评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 166,090评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,010评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,054评论 6 395
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,806评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,484评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,380评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,873评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,021评论 3 338
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,158评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,838评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,499评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,044评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,159评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,449评论 3 374
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,136评论 2 356

推荐阅读更多精彩内容