Python Iterator and Generator
Python Iterator and Generator
Iterator
迭代器(Iterator)和可迭代对象(Iterable)往往是绑定的。可迭代对象就是我们平时经常用的list
,string
, tuple
这种。事实上迭代器的概念会比可迭代对象宽泛很多,一会举几个例子就能明白。
在使用list
这种数据类型的时候,我们经常会使用下面这种迭代方式:
# eg 1
mylist = [1,2,3,4]
for x in mylist:
print(x)
>>>1
>>>2
>>>3
>>>4
有时候会很奇怪for循环为什么可以这么用,列表的索引竟然会自动往后一个一个走,走到结尾了,还会自动停下来,不会有list out of range
的报错。神奇。其实for循环做的事情不止这么简单。要说明这个问题,得先说迭代器具体怎么做,是什么。
要创建一个迭代器,就必须要实现两个方法,分别是__iter__()
和__next__()
,以及实现异常机制StopIteration
。请看下面的例子:
# eg 2
class PowTwo:
"""Class to implement an iterator of powers of two"""
def __init__(self, max = 0):
self.max = max
def __iter__(self):
self.n = 0
return self
def __next__(self):
if self.n <= self.max:
result = 2 ** self.n
self.n += 1
return result
else:
raise StopIteration
可以看到,迭代器是通过写一个类来定义的,类里面实现了刚刚所说的__iter__()
和__next__()
方法。这个迭代器是用来产生一系列2的指数次方的数字的,具体用法看下面:
a = PowTwo(4)
i = iter(a) # attention please
next(i)
>>>1
next(i)
>>>2
next(i)
>>>4
next(i)
>>>8
next(i)
>>>16
next(i)
>>>Traceback (most recent call last):
...
StopIteration
仔细看哦,第一行代码用4创建了一个实例a
,设置了这个迭代器的迭代上限,然后并不是直接用这个实例就可以了,还得调用iter()
函数去把a
彻底进化成一个Iterator
,进而有了接下来next()
函数的闪亮登场。其实我也蛮奇怪,直接用这个类不好么,比如下面的代码:
a = PowTwo(4)
a.__iter__()
a.__next__()
>>>1
a.__next__()
>>>2
a.__next__()
>>>4
a.__next__()
>>>8
a.__next__()
>>>16
next(i)
>>>Traceback (most recent call last):
...
StopIteration
完全没问题,但是你自己比较一下二者的代码,哪个美一点毋庸置疑。。。再说了,跟装饰器一样,一切为了简洁优美而生嘛。
OK,可以回到最初的起点——for循环了。示例1中的代码,for循环到底做了什么呢,答案是,for循环其实是先把mylist
变成迭代器,然后用while循环去迭代:
iter_obj = iter(mylist)
while True:
try:
element = next(iter_obj)
except StopIteration:
break
这样一路解释过来,应该就不难理解迭代器了。总结一下就是:
- 如何已经有一个可迭代对象,那么直接用
iter()
函数去初始化它,然后疯狂next()
即可; - 如果没有现成的可迭代对象,那就自己写一个类,类里面记得实现
__iter__()
和__next__()
方法,以及异常机制StopIteration
,然后操作同1; - 如果你想要一个无限迭代器,不要实现异常机制
StopIteration
即可。
Generator
这玩意儿就比迭代器复杂点了,所以还得分几个小点,逐个击破。
1. 生成器是啥
生成器也是Python中面向一系列需要迭代的问题,常用的解决方案。既然有了迭代器,可以解决很多迭代的问题,那为啥子还要生成器勒?
主要的原因是迭代器的开销太大了。对于一些小问题还好,大的问题需要迭代的元素很庞大的时候,迭代器就使不上劲儿了。而且,创建一个迭代器,说实话也还挺麻烦的,看看上面的小总结的第二点,你得实现这些方法和手动处理异常。
而且迭代器要写类,其实一个函数可以搞定的事情,何必那么复杂。正是应了这个景,生成器也就是在函数对象上搞事情的。
这个原因点到即止,先把生成器讲清楚了,自然就通透了。先看一个小例子,写一个生成器函数(Generator Function):
# eg 1
def my_gen():
n = 1
print('This is printed first')
yield n
n += 1
print('This is printed second')
yield n
n += 1
print('This is printed at last')
yield n
上面就是一个简单的生成器函数,多了一种关键字yield
,细心看会发现这个函数竟然没有return
!再细看代码,没错,yield
替代了return
。那怎么用呢,有两种用法如下:
# usage 1
a = my_gen()
next(a)
>>>This is printed first
1
next(a)
>>>This is printed second
2
next(a)
>>>This is printed at last
3
next(a)
>>>Traceback (most recent call last):
...
StopIteration
# usage 2
for item in my_gen():
print(item)
>>>
This is printed first
1
This is printed second
2
This is printed at last
3
对于用法1,把函数赋值给了a
,然乎疯狂next()
即可。你会发现,我们并没有像迭代器那样实现__iter__()
和__next__()
方法,以及异常机制StopIteration
,只是用了一个yield
关键字,这个生成器函数却达到了迭代器一样的效果。
对于用法2,更牛皮了,甚至不用赋值的操作,直接for这个生成器函数。。。
2. 循环机制生成器
刚刚那个小函数,就是一个最普通的例子,那问题是如果有多个n想要玩,岂不是来多少手动写多少?那当然还有循环的玩法,带有循环机制的生成器。下面是一个逆序输出的小例子:
# eg 2
def rev_str(my_str):
length = len(my_str)
for i in range(length - 1,-1,-1):
yield my_str[i]
for char in rev_str("hello"):
print(char)
>>>o
>>>l
>>>l
>>>e
>>>h
没错,真无聊,犯得着逆序输出的程序还得上生成器么,犯不着,但是,只是想给出这个循环机制生成器的概念。如果你发现这个rev_str()
函数和普通的逆序函数完全一样,只是return
换成了yield
,那就万事大吉,要理解的就是这个。就这样一记简单的替换操作,你就得到了一个生成器,是不是比迭代器省事儿多了。
3. 生成器表达式(Generator Expression)
不知道你有没有用过列表生成式,没用过也应该看到过,这类似于匿名函数,语法简洁,比如:
# eg 3
my_list = [1, 3, 6, 10]
[x**2 for x in my_list]
>>>[1, 9, 36, 100]
生成器表达式和这个几乎一样,不信你看:
# eg 4
my_list = [1, 3, 6, 10]
a = (x**2 for x in my_list)
next(a)
>>>1
next(a)
>>>9
next(a)
>>>36
next(a)
>>>100
把列表生成式的[]
直接改成()
,就得到了一个生成器。
4. Why Generator???
(1). 简洁
回到最开始迭代器的那个类的例子,用生成器咋写呢?
def PowTwoGen(max = 0):
n = 0
while n < max:
yield 2 ** n
n += 1
简洁吧,然后你就可以用这个生成器函数遨游了。
(2). 开销小
同样的一个需要迭代的功能,如果用普通函数写,一旦需要迭代的元素特别多,在使用的时候,普通函数需要等所有的元素计算出来了,然后把返回值给你。生成器就不是了,它一次计算出一个,用的时候就取一个,并且它还会记住位置,下次用就计算下一个,这样对空间的开销也是很小的。
(3). 无限
看下面的函数:
def all_even():
n = 0
while True:
yield n
n += 2
写普通函数,你必然做不到写出一个可以无限操作的函数,生成器却可以。(迭代器也可以,就是麻烦点儿)
5. 再给一个例子
# 利用yield生成一个斐波那契数列的生成器
def fib(max):
n,a,b=0,0,1
while n<max:
yield b
a,b=b,a+b
n+=1
return 'done' # 要不要这句话都行
f=fib(6)
next(f) # 疯狂next()它
>>>
1
1
2
3
5
8
Traceback (most recent call last):
...
StopIteration: done
Python Iterator and Generator的更多相关文章
- [Python学习]Iterator 和 Generator的学习心得
[Python学习]Iterator 和 Generator的学习心得 Iterator是迭代器的意思,它的作用是一次产生一个数据项,直到没有为止.这样在 for 循环中就可以对它进行循环处理了.那么 ...
- Python3 Iterator and Generator
Python3 Iterator and Generator iterator 主要是利用 iter 函数 >>> list=[1,2,3,4] >>> it = ...
- babel 不能统编译Iterator、Generator、Set、Maps、Proxy、Reflect、Symbol、Promise的问题
Babel默认只转换新的JavaScript句法(syntax),而不转换新的API,比如Iterator.Generator.Set.Maps.Proxy.Reflect.Symbol.Promis ...
- Python生成器(generator)和迭代器(Iterator)
列表生成式 a = [i+1 for i in range(10)] print(a) [1, 2, 3, 4, 5, 6, 7, 8, 9, 10] 这就是列表生成式 生成器(generator) ...
- Python之生成器(generator)和迭代器(Iterator)
generator 生成器generator:一边循环一边计算的机制. 生成器是一个特殊的程序,可以被用于控制循环的迭代行为.python中的生成器是迭代器的一种,使用yield返回值函数,每次调用y ...
- python中的generator, iterator, iterabel
先来看看如果遇到一个对象,如何判断其是否是这三种类型: from types import GeneratorType from collectiuons import Iterable, Itera ...
- Python中生成器generator和迭代器Iterator的使用方法
一.生成器 1. 生成器的定义 把所需要值得计算方法储存起来,不会先直接生成数值,而是等到什么时候使用什么时候生成,每次生成一个,减少计算机占用内存空间 2. 生成器的创建方式 第一种只要把一个列表生 ...
- python中的generator(coroutine)浅析和应用
背景知识: 在Python中一个function要运行起来,它在python VM中需要三个东西. PyCodeObject,这个保存了函数的代码 PyFunctionObject,这个代表一个虚拟机 ...
- (转)Python中的generator详解
本文转自:http://www.cnblogs.com/xybaby/p/6322376.html 作者:xybaby 注:本文在原文基础上做了一点点修改,仅仅作为个人理解与记忆,建议直接查看原文. ...
随机推荐
- spark 源码分析之五 -- Spark内置RPC机制剖析之一创建NettyRpcEnv
在前面源码剖析介绍中,spark 源码分析之二 -- SparkContext 的初始化过程 中的SparkEnv和 spark 源码分析之四 -- TaskScheduler的创建和启动过程 中的C ...
- 阿里巴巴 -- MySQL DBA 面试题
1.MySQL的复制原理以及流程 (1).先问基本原理流程,3个线程以及之间的关联: (2).再问一致性延时性,数据恢复: (3).再问各种工作遇到的复制bug的解决方法. 2.MySQL中myisa ...
- Anemic Model
In object-oriented programming, and especially in Domain-Driven Design, objects are said to be anemi ...
- C# 使用Quartz简单实例以及备忘
一.导入NuGet 二.创建一个类并实现接口Ijob,并实现该接口中的方法. using Buday.Gold.Cusumer.Lib; using Quartz; using System; us ...
- Contiki源码分析--CPU为cc253x里的uart0.c
我所使用的Contiki系统是contiki-sensinode.理解该文需要有cc2530里uart的相关知识,具体寄存器的用法不做介绍. 先放上所有代码,然后再仔细分析. #include < ...
- Excel中vlookup函数使用
https://baijiahao.baidu.com/s?id=1594684818733205984&wfr=spider&for=pc
- cogs.12运输问题2题解
乍一看貌似和运输问题1没有任何区别,但本题有一个有意思的东西叫做下限,我个人称之为非强制下限,因为本题中要求的实际是我走这条边这条边才至少走下限的流,虽然出题人没说,但从样例来看确实是这样的,而强制下 ...
- Bzoj 1040 [ZJOI2008]骑士 题解
1040: [ZJOI2008]骑士 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 5368 Solved: 2044[Submit][Status ...
- 关键字static、final
final final能修饰类.修饰方法.能修饰属性. 修饰类:该类不能被继承. 修饰方法:该方法不能被重写.所以abstract和final不能同时用 修饰属性/变量:该属性/变量为常量,该值不能再 ...
- hdu6406 Taotao Picks Apples(线段树)
Taotao Picks Apples 题目传送门 解题思路 建立一颗线段树,维护当前区间内的最大值maxx和可摘取的苹果数num.最大值很容易维护,主要是可摘取的苹果数怎么合并.合并左右孩子时,左孩 ...