Python基础之迭代器、生成器

这部分理解起来比较费劲,多练习,多思考~

一 生成器

我们知道列表的大小是有限制的,假设我们要放自然数0~100万,那这个列表肯定占了很大内存空间,而且我们也可能只用到其中很小一部分的数据,这样的话我们用列表造成的浪费就很大了。但是,如果这个列表的元素是有规律可循的,那我们就可以根据这个规律,推算到任何一个元素,不必创建完整的list,从而节省大量的空间。在Python里,这种一边循环一边计算的机制,称为生成器generator。

  1. 第一种简单的generator,是把列表生成式的[ ] 改成()即可
L = [i * i for i in range(1,11) if i%2==0]
print(L) # [4, 16, 36, 64, 100]

g=(i * i for i in range(1,11) if i%2==0)
print(g) # <generator object <genexpr> at 0x000002D6BDBADBF8>

通过next(g)可以打印出generator中的元素:

print(next(g))  #4
print(next(g))  #16
print(next(g))  #36
print(next(g))  #64
print(next(g))  #100
# print(next(g)) #StopIteration 没有足够的元素,调用next会抛出StopIteration。

当然更简单的是通过for循环遍历generator,同时还防止了StopIteration:

g=(i * i for i in range(1,11) if i%2==0)
for i in  g:
    print(i)

输出:

4
16
36
64
100

能用for遍历,说明generator是个可迭代对象。

  1. 当规律更复杂,不能用列表生成式那样简单生成,可以用带yield 的函数来生成。一旦函数带了yield 关键字,那这个函数就不是通常的函数了,而是generator。
def func():
    print('step 1')
    yield 1
    print('step 2')
    yield (2)
    print('step 3')
    yield (3)

f=func()
print(next(f))
print(next(f))
print(next(f))

看下输出,分析下结果:

step 1
1
step 2
2
step 3
3

第一次执行next(f),函数内部打印了 step 1,再执行了 yield 1。而且这个1,整好被我们print打印出来了。也就是说next(f)返回了1,结果上类似于return 1 了,此时函数终止在这里。
第二次执行next(f),函数继续往下走,先打印step 2,再执行yield (2),函数终止,返回 2,并打印。
第三次执行next(f),函数继续往下走,先打印step 3,再执行yield (3),函数终止,返回 3,并打印。
可以得出结论:带yield的generator,遇到yield函数终止,并返回相应的值。这就是规则。
等同于用for来遍历:

f=func()
for i in f:
    print(i)

运行结果完全一样。
好像有点乱~对于一个generator来说,用for遍历的效果等同于不停的调用next(忽略异常)。
到这里,我们只需要清楚生成器是什么(一边循环一边计算的机制),他的两种常见形式,如何打印元素,执行流程即可。

二 迭代器

像上面讲到的generator,可以被next()函数调用并不断返回下一个值的对象称为迭代器:Iterator。

除此之外,像集合数据类型,如list、tuple、dict、set、str等,可以直接作用于for循环的对象统称为可迭代对象:Iterable。generator也可以直接作用于for,所以generator同时也是Iterable。

生成器都是Iterator对象,但list、dict、str虽然是Iterable,却不是Iterator。因为Python的Iterator对象表示的是一个数据流,通过next()函数调用并不断返回下一个数据。所以可以把他看成一个有序的,未知大小的序列,可以看成无限大,而我们常见的集合型对象都是有限大小的。

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

推荐阅读更多精彩内容