同学们,大家好。鉴于昨天拖更了一天,所及今天要讲一些浓缩的精华的很重要知识点,即python中十分重要的迭代器、生成器和列表解析,好了闲话不多说,进入正题。
1、首先介绍一下迭代器,初学者对这个概念可能有一些陌生,在理解上也有些难度,我在这里详细介绍一下。
首先,我们要明白什么是迭代器。在说这个概念之前,我们还有一个概念需要知道,即可迭代对象——简单的说就是可以使用for...in...语句进行循环的对象,比如字符串、列表、元组、字典、文件等。
那么什么是迭代器呢?
即可以被_next_()函数调用并不断返回下一个值的对象称为迭代器。
那么迭代器和可迭代对象是什么关系呢?
即可迭代对象可以通过iter()函数转化为迭代器。
所以我们也可以得出一个结论,即迭代器一定是迭代对象,迭代对象不一定是迭代器。
这么说对初学者可能太抽象了,下面我们举一个例子。
注:判断一个对象是否是可迭代对象可以调用collections模块里面的iterable包的isinstance函数进行判断。 例如:isinstance('abc',Iterable) 意思是字符串abc是可迭代对象吗?
a=[1,2,3]
#列表a是一个可迭代对象,因为列表a可以使用for循环遍历,所以a是一个可迭代对象。
此时我们需要把可迭代对象转换为一个迭代器,上文中说过,即把列表a传入iter()函数。
b=iter(a)#此时a就成为一个迭代器了,此时可以调用next()函数。
next(b)#此时打印出1
next(b)#此时打印出2
next(b)#此时打印出3
next(b)#迭代器耗尽,出现StopIteration
以上便是一个迭代器最最基本的例子,那么就有人问了,使用迭代器的好处是什么呢? 根据我的理解,迭代器可以加快运行速度,优化代码结构,节省内存,比如读取一个文件,我们可以不必一次性读取整个文件,二是可以使用跌大气逐行读取,后者的运行速度是前者的2倍。
2、列表解析。
列表解析写在一个方括号里面,具体句式对列表内的元素进行相同的操作,几个例子:
首先如果给我们一个列表a=[1,2,3]
如果我们想让列表中的每个元素乘以2,一般会构造如下代码:for i in a:
a[i]=a[i]*2
如果使用列表解析呢,一行代码足够:
b=[i*2 for i in a]
上述两种操作的结果是一样的,但是第二种相对更简洁,此即为列表解析。具体的 格式如下:
[句子 for 变量 in 可迭代对象]
首先列表解析必须写在一对方括号之中,句子指的是你要对列表中的每一个元素进行的操作;变量指的是将列表中的元素每次迭代都赋值给变量。
我们再举一例,例如将列表a中所有的元素加10,可以这么写:
b=[i+10 for i in a]
3、两个重要的函数
一个是map函数一个事zip函数,这两个函数在以后的编程中会常用,需牢记。
map函数接收两个参数,一个是函数,一个事序列,并将序列中的每个元素传入到函数,并返回迭代器下面举例说明:
列表a=[-1,-3,1]
如果我们想对列表a中的每一个元素取绝对值,除了上述的列表解析,可以用map函数:
m=map(abs,a)
zip函数的作用则是接收两个序列,将其对应组成一对元组,并返回迭代器。举例如下:
a=[1,2,3]
b=[4,5,6]
c =zip(a,b)
list(c)
#结果为[(1,4),(2,5),(3,6)]
3、生成器。
首先我们要知道,生成器是一种边循环边计算的机制。
创建一个生成器generator,有很多种方法。一种方法是个列表生成式的[]改成(),就创建了一个generator:
L = [x * 2 for x in range(3)]
L [0, 1, 4]
g = (x * 2 for x in range(3))
g at 0x104feab40>
第二种方法是使用yield关键字,举例如下:
def a gen(N)
for i in range(N):
yield i**2
此函数在每次循环时生成一个值,然后停止,挂起这个状态,然后再调用,再挂起,直到耗尽为止。
那么生成器的作用是什么呢?
假如我们要建立一个包含500万个元素的列表,这会占掉很大内存,但是我们只想要前三个元素的值,那么后面庞大的元素所占的空间不就浪费了么,所以此时需要生成器,变循环边计算,调出前三个值。
注:生成器自身便是迭代器 ,支持迭代器的一切操作。