Python 进阶_迭代器 & 列表解析
目录
迭代器
迭代器是一个含有 next()
方法的对象,让我们可以迭代不是序列数据类型但表现出序列行为的对象,所以可以说迭代器为类序列对象提供了一个类序列的接口(只要是实现了 __iter__()
方法的对象,就可以使用迭代器来进行访问)。迭代器从对象的第一个元素开始访问,直到所有的元素被遍历后结束。对于无法通过索引计数来随机访问元素的数据结构(EG. set)而言,迭代器是唯一的访问其自身元素的方式。
NOTE1: 但迭代器是不支持索引计数的,所以迭代器不能回退,只能往前进行迭代。
NOTE2: 迭代器也不是线程安全的,在多线程环境中对可变对象使用迭代器是一个危险的操作。所以一般情况下应该坚持对不可变对象实现迭代器。
NOTE3: 迭代器对象不支持被多次迭代
In [19]: a = ListIter([1,2,3,4,5])
In [20]: [i for i in a]
Out[20]: [1, 2, 3, 4, 5]
In [21]: [i for i in a]
Out[21]: []
In [22]:
iter() :内建的迭代器生成函数
iter(…)
iter(collection) -> iterator
iter(callable, sentinel) -> iterator
Get an iterator from an object. In the first form, the argument must
supply its own iterator, or be a sequence.
In the second form, the callable is called until it returns the sentinel.
1. 如果传递了一个序列(sequence)实参,迭代器会从索引 0 一直迭代至结束。
2. 如果传递了两个实参 EG. iter(func, sentinel)
,迭代器会重复的调用 func 直到迭代器的下一个值为 sentinel 。
EXAMPLE 1:
In [44]: aList = ['jmilkfan', 'fanguiju', 'chocolate']
In [45]: aIter = iter(aList)
In [46]: aIter.next()
Out[46]: 'jmilkfan'
In [47]: aIter.next()
Out[47]: 'fanguiju'
In [48]: next(aIter)
Out[48]: 'chocolate'
In [49]: aIter.next()
---------------------------------------------------------------------------
StopIteration Traceback (most recent call last)
<ipython-input-49-5ab62b0e2847> in <module>()
----> 1 aIter.next()
StopIteration:
迭代器通过其内建的 iter.next()
方法,或通过 Python 内建的 next()
来迭代下一个元素,直到最后触发 StopIteration
异常后表示迭代结束。
EXAMPLE 2: 捕获异常
In [56]: aList = ['jmilkfan', 'fanguiju', 'chocolate']
In [57]: aIter = iter(aList)
In [58]: while True:
...: try:
...: print aIter.next()
...: except StopIteration:
...: print 'Done'
...: break
...:
jmilkfan
fanguiju
chocolate
Done
迭代器在 for 循环中
EXAMPLE 3: 对 EXAMPLE 2 的改进
In [59]: aList = ['jmilkfan', 'fanguiju', 'chocolate']
In [60]: aIter = iter(aList)
In [61]: for x in aIter:
...: print x
...:
jmilkfan
fanguiju
chocolate
Python 在 for 循环的语法糖中,让 for 循环能够自动的调用迭代器的 next()
方法以及捕获 StopIteration
异常。
迭代器与字典
字典是一个可迭代对象,其迭代器会变量它的 key, 所以我们可以应用这个特性将语句 for eachKey in myDict.keys()
改进为 for eachKey in myDict
EXAMPLE 4:
In [62]: aDict = {'name':'jmilkfan', 'sex':'man'}
In [69]: for x in aDict:
...: print ''.join([x,': ',aDict[x]])
...:
name: jmilkfan
sex: man
迭代器与文件
文件也是一个可迭代对象,迭代器会自动的调用文件对象的 readline()
方法,所以可以将语句 for eachLine in myFile.readlines()
修改为 for eachLine in myFile
EXAMPLE 5:
myFile = open('FILENAME')
for eachLine in myFile:
print eachLine
myFile.close()
创建迭代器对象
EXAMPLE 6: 一个斐波那契数列
class Fab(object):
def __init__(self, max):
self.max = max
self.n, self.a, self.b = 0, 0, 1
def __iter__(self):
return self
def next(self):
if self.n < self.max:
r = self.b
self.a, self.b = self.b, self.a + self.b
self.n = self.n + 1
return r
raise StopIteration()
if __name__ == '__main__':
fab = Fab(5)
print fab
for x in fab:
print x
Output:
In [79]: run demo_1.py
<__main__.Fab object at 0x00000000047CDE80>
1
1
2
3
5
将实例化语句 fab = Fab(5)
修改成 fab = Fab(100)
后执行,再看看 Output:
1
.
.
.
354224848179261915075
NOTE: 这一类的应用场景会对内存造成非常大的负担,建议使用迭代器来减少内存的压力。
EXAMPLE 6 中的 __iter__()
方法 return 了自己,所以迭代的对象就是自身。除此之外,我们还可以实现 委托迭代。
创建迭代对象并实现委托迭代
委托迭代:就是将迭代请求委托到迭代对象内部持有的容器对象上。它能让自己创建的新容器能够完成迭代操作。
EXAMPLE 7:
class Node:
def __init__(self, value):
self._value = value
self._children = [] # 迭代对象所持有的容器对象
def __repr__(self):
return 'Node({!r})'.format(self._value)
def add_child(self, node):
self._children.append(node)
def __iter__(self):
return iter(self._children) # 将迭代请求转发给迭代对象内部所持有的容器对象
if __name__ == '__main__':
root = Node(0)
child1 = Node(1)
child2 = Node(2)
root.add_child(child1)
root.add_child(child2)
for ch in root: # 调用 Node:__iter__()
print(ch) # 调用 Node:__repr__()
Output:
Node(1)
Node(2)
EXAMPLE 7 是一个树结构,执行自身(root),返回子节点(Node(1)、Node(2)) 。这个例子,当我们 for 循环遍历迭代器对象 root 时,实际上是迭代了 self._children = [Node(1), Node(2)]
,这就是迭代委托。
迭代器的多次迭代
在上文写到,迭代器不支持被多次迭代,这样实在不能说是灵活。为了解决这一个问题,引入了可迭代对象(iterables)和迭代器对象(iterator)两个不同的概念。
- 迭代器对象:
__iter__()
返回的是迭代器对象自身。 - 可迭代对象:
__iter__()
返回了一个迭代器对象。
上述的 委托迭代 就是一个返回一个迭代器对象的例子,所以 EXAMPLE 7 是一个 可迭代对象,他能够被多次迭代。
In [1]: run demo_1.py
Node(1)
Node(2)
In [2]: run demo_1.py
Node(1)
Node(2)
列表解析
是一个非常有用、简单且灵活的工具,让我们能够动态的创建列表类型对象。
语法:
[expr for iter_variable in iterable]
EXAMPLE 1:
In [18]: [x**2 for x in range(10)]
Out[18]: [0, 1, 4, 9, 16, 25, 36, 49, 64, 81]
In [80]: map(lambda x:x ** 2, range(10))
Out[80]: [0, 1, 4, 9, 16, 25, 36, 49, 64, 81]
上面两条语句的效果是一样的,但列表解析的方法仅调用一次 range(10)
, 而第二条则调用了 map()/lambda/range(10)
,这说明列表解析可以替代 map() 以及 lambda ,以此来获取更高的效率。
列表解析的样例
- 嵌套 for 循环实现的矩阵
In [81]: [(x+1, y+1) for x in range(3) for y in range(5)]
Out[81]:
[(1, 1),
(1, 2),
(1, 3),
(1, 4),
(1, 5),
(2, 1),
(2, 2),
(2, 3),
(2, 4),
(2, 5),
(3, 1),
(3, 2),
(3, 3),
(3, 4),
(3, 5)]
- 统计文件单词数
f = opem('FILENAME', 'r')
len([word for line in f for word in line.split()])
- 求素数
[x for x in range(2,100) if not [y for y in range(2,int(x/2+1)) if x % y == 0]]
- 嵌套列表降维
In [143]: nestLi = [[1,2,3],[4,5,6],[7,8,9]]
In [144]: newLi = [x for para in nestLi for x in para]
In [145]: newLi
Out[145]: [1, 2, 3, 4, 5, 6, 7, 8, 9]
列表解析和迭代器
因为 for 循环的关系,所以从上面的例子可以看出列表解析和迭代器之间的关系非常紧密。深入的理解两者,对提高 Python 程序的效率有非常大的帮助。迭代是 Python 一个非常重要的思想和特性。
Python 进阶_迭代器 & 列表解析的更多相关文章
- Python 进阶_生成器 & 生成器表达式
目录 目录 相关知识点 生成器 生成器 fab 的执行过程 生成器和迭代器的区别 生成器的优势 加强的生成器特性 生成器表达式 生成器表达式样例 小结 相关知识点 Python 进阶_迭代器 & ...
- python利用or在列表解析中调用多个函数.py
python利用or在列表解析中调用多个函数.py """ python利用or在列表解析中调用多个函数.py 2016年3月15日 05:08:42 codegay & ...
- python进阶_浅谈面向对象进阶
python进阶_浅谈面向对象进阶 学了面向对象三大特性继承,多态,封装.今天我们看看面向对象的一些进阶内容,反射和一些类的内置函数. 一.isinstance和issubclass class F ...
- Python入门笔记(13):列表解析
一.列表解析 列表解析来自函数式编程语言(haskell),语法如下: [expr for iter_var in iterable] [expr for iter_var in iterable i ...
- Python进阶:迭代器与迭代器切片
2018-12-31 更新声明:切片系列文章本是分三篇写成,现已合并成一篇.合并后,修正了一些严重的错误(如自定义序列切片的部分),还对行文结构与章节衔接做了大量改动.原系列的单篇就不删除了,毕竟也是 ...
- Python基础:08列表解析与生成器表达式
一:列表解析 列表解析(List comprehensions)来自函数式编程语言Haskell .它可以用来动态地创建列表.它在 Python 2.0 中被加入. 列表解析的语法: [exp ...
- Python进阶-V 迭代器(Iterator)、生成器(Generator)函数
一.迭代器 1.可循环的有哪些,即可用for语句或者while语句的数据类型有哪些? 字符串(str).列表(list).元组(tuple).字典(dic).集合(set).枚举类(enumerate ...
- Python 进阶_模块 & 包
目录 目录 模块的搜索路径和路径搜索 搜索路径 命名空间和变量作用域的比较 变量名的查找覆盖 导入模块 import 语句 from-import 语句 扩展的 import 语句 as 自动载入模块 ...
- python 进阶篇 迭代器和生成器深入理解
列表/元组/字典/集合都是容器.对于容器,可以很直观地想象成多个元素在一起的单元:而不同容器的区别,正是在于内部数据结构的实现方法. 所有的容器都是可迭代的(iterable).另外字符串也可以被迭代 ...
随机推荐
- 20190817 On Java8 第七章 封装
第七章 封装 访问控制权限的等级,从"最大权限"到"最小权限"依次是:public,protected,包访问权限(没有关键字)和 private. 包的概念 ...
- HttpUrlConnection工具类
package com.ligotop.core.utils; import com.ligotop.core.exceptions.BusinessException; import java.io ...
- 二叉树BinTree4种遍历及其应用
前序遍历 template<class T> void BinTree<T>::PreOrder(BinTreeNode<T>*subTree){ //前序遍历以s ...
- Codeforces 1080C 题解(思维+二维前缀和)
题面 传送门 题目大意: 有一个黑白的棋盘,现在将棋盘上的一个子矩形全部染成黑色,另一个子矩形全部染成白色 求染完色后黑,白格子的总数 分析 我们可以发现,对于一个(1,1)到(x,y)的矩形,若xy ...
- Matplotlib基础使用
matplotlib 一.Matplotlib基础知识 Matplotlib中的基本图表包括的元素 x轴和y轴 axis 水平和垂直的轴线 x轴和y轴刻度 tick 刻度标示坐标轴的分隔,包括最小刻度 ...
- 17、NumPy——副本和视图
副本是一个数据的完整的拷贝,如果我们对副本进行修改,它不会影响到原始数据,物理内存不在同一位置. 视图是数据的一个别称或引用,通过该别称或引用亦便可访问.操作原有数据,但原有数据不会产生拷贝.如果我们 ...
- JS面向对象——动态原型模型、寄生构造模型
动态原型模型 组合使用构造函数模型和原型模型,使得OO语言程序员在看到独立的构造函数和原型时很困惑.动态原型模型致力于解决该问题,它把所有的信息封装在构造函数中,通过在构造函数中初始化原型(仅在必要情 ...
- 四、局域网连接SqlServer
一.局域网连接SqlServer 一台服务器上装有四个数据库的时候,我们可以通过IP\实例名的方式进行访问. navicat 连接sqlserver数据库
- 《深入学习Redis(2):持久化》笔记
参考 https://www.cnblogs.com/kismetv/p/9137897.html 一.高可用概述 提供正常服务:主从分离,快速容灾技术,数据容量的扩展.数据安全不会丢失. 持久 ...
- IText PDF简单示例
package com.exe.learn.demo.itextpdf; import java.io.ByteArrayInputStream; import java.io.File; impor ...