一、Python的迭代协议

  迭代器是访问集合内元素的一种方式。它只能依次访问集合内元素。其特点是惰性执行。

  collection.abc的迭代协议提供了两个概念:可迭代对象和迭代器。可迭代对象:必须具有__item__特殊方法;迭代器:必须具有__next__方法。

class Iterable(metaclass=ABCMeta):
__slots__ = ()
@abstractmethod
def __iter__(self):
while False:
yield None
@classmethod
def __subclasshook__(cls, C):
if cls is Iterable:
return _check_methods(C, "__iter__")
return NotImplemented class Iterator(Iterable):
__slots__ = ()
@abstractmethod
def __next__(self):
'Return the next item from the iterator. When exhausted, raise StopIteration'
raise StopIteration
def __iter__(self):
return self # 重载了Iterable的__iter__
@classmethod
def __subclasshook__(cls, C):
if cls is Iterator:
return _check_methods(C, '__iter__', '__next__')
return NotImplemented Iterator.register(bytes_iterator)
Iterator.register(bytearray_iterator)
#Iterator.register(callable_iterator)
Iterator.register(dict_keyiterator)
Iterator.register(dict_valueiterator)
Iterator.register(dict_itemiterator)
Iterator.register(list_iterator)
Iterator.register(list_reverseiterator)
Iterator.register(range_iterator)
Iterator.register(longrange_iterator)
Iterator.register(set_iterator)
Iterator.register(str_iterator)
Iterator.register(tuple_iterator)
Iterator.register(zip_iterator)

  从上面可以看到,字节、字典、列表、集合、元组、字符串和range、zip都被注册到了迭代器元类中,成为可迭代对象。实际上,open文件句柄也是一个迭代器。

二、可迭代对象与迭代器

  1.可迭代对象

  对象含有__iter__方法,就是可迭代对象。不含__iter__方法,也可能是可迭代对象。__getitem__方法会自动创建__iter__来满足遍历条件。

class Person:
def __init__(self, persons):
self.persons = persons
# def __iter__(self):
# return 1
def __getitem__(self, item): # 假如__iter__没有设定,解释器会继续查找__getitem__
return self.persons[item] # __getitem__会自动创建__iter___来进行迭代 per = Person(["Bob", "Tom", "Li", "Jan"])
print(per[:3])
for p in per:
print(p) """
['Bob', 'Tom', 'Li']
Bob
Tom
Li
Jan
"""

  2.迭代器

  iter()用来将可迭代对象生成迭代器。

class Person:
def __init__(self, persons):
self.persons = persons
def __getitem__(self, item):
return self.persons[item]
return len(self.persons) per = Person(["Bob", "Tom", "Li", "Jan"])
my = iter(per)
for i in range(len(per)):
print(next(my)) """
Bob
Tom
Li
Jan
"""

  3.迭代器的实现

  当需要定义可迭代对象时,一般单独写一个迭代器功能。不建议在可迭代对象内写__next__

class MyIterator:
def __init__(self, persons):
self.persons = persons
self.__index = 0
def __iter__(self):
return self
def __next__(self):
try:
per = self.persons[self.__index]
except IndexError:
raise StopIteration
self.__index += 1
return per class Person:
def __init__(self, persons):
self.persons = persons
def __len__(self):
return len(self.persons) def __iter__(self):
return MyIterator(self.persons) # 当要定义可迭代对象时,通常不在类内写__next__,而是写在外面 per = Person(["Bob", "Tom", "Li", "Jan"])
my = iter(per)
for i in range(len(per)):
print(next(my))

三、生成器

  当函数中存在yield时,它就不再是普通的函数了,而是一个生成器函数。

def f1():
print("-----------step1----------")
yield 1
print("-----------step2----------")
yield 2 g = f1() # 这一步没执行
print(next(g))
print(next(g))

  生成器的实现,本质上是对函数的堆存储方式进行了一层封装。

  普通函数的调用和执行过程。被调用函数独在字节码这一步单独存储于PyCodeObject中。它必须一次性运行完,而不会保留执行时的栈帧。

    

import inspect
frame = None
def foo():
bar()
def bar():
global frame
frame = inspect.currentframe() foo()
print(dis.dis(foo))
print(frame.f_code.co_name)
print(frame.f_back.f_code.co_name, frame.f_back) # 所有的栈帧都是分配在对内存当中,独立于调用者存在

PyCodeObject

   生成器对象(PyGenObject)会被编译成相应的字节码,以及每一次yield的栈帧和本地变量。相比于普通函数,PyGenObject多了对f_lasti和f_locals的保存。只要每次yield拿到f_lasti和gen_fn's bytecode,就可以暂停或继续执行生成器。

    

import dis
def f1():
print("-----------step1----------")
name = "Li"
yield 1
print("-----------step2----------")
age = 24
yield 2 g = f1() # 这一步没执行
print(g.gi_code) # 显示代码的起止位置
print(dis.dis(g)) # 显示栈帧与字节码
print(g.gi_frame.f_lasti, g.gi_frame.f_locals) # 显示代码执行至此的栈帧和本地变量集
print(next(g))
print(g.gi_frame.f_lasti, g.gi_frame.f_locals)
print(next(g))
print(g.gi_frame.f_lasti, g.gi_frame.f_locals) """
<code object f1 at 0x110b71f60, file "<ipython-input-261-13e8f3cdc8da>", line 2>
3 0 LOAD_GLOBAL 0 (print)
2 LOAD_CONST 1 ('-----------step1----------')
4 CALL_FUNCTION 1
6 POP_TOP 4 8 LOAD_CONST 2 ('Li')
10 STORE_FAST 0 (name) 5 12 LOAD_CONST 3 (1)
14 YIELD_VALUE
16 POP_TOP 6 18 LOAD_GLOBAL 0 (print)
20 LOAD_CONST 4 ('-----------step2----------')
22 CALL_FUNCTION 1
24 POP_TOP 7 26 LOAD_CONST 5 (24)
28 STORE_FAST 1 (age) 8 30 LOAD_CONST 6 (2)
32 YIELD_VALUE
34 POP_TOP
36 LOAD_CONST 0 (None)
38 RETURN_VALUE
None
-1 {}
-----------step1----------
1
None
14 {'name': 'Li'}
-----------step2----------
2
32 {'name': 'Li', 'age': 24}
"""

PyGenObject

  待续。。。。。。

python(九):迭代器与生成器的更多相关文章

  1. python(九)迭代器和生成器

    一.迭代 迭代就是逐个并且单向访问容器 (这里的容器暂时指数据类型,比如list和dict) 中的元素的行为.举个例子:将一个长度为五的数组逐个从头到尾(即单向)打印的方式称之为迭代.如下图. > ...

  2. python基础—迭代器、生成器

    python基础-迭代器.生成器 1 迭代器定义 迭代的意思是重复做一些事很多次,就像在循环中做的那样. 只要该对象可以实现__iter__方法,就可以进行迭代. 迭代对象调用__iter__方法会返 ...

  3. python之迭代器与生成器

    python之迭代器与生成器 可迭代 假如现在有一个列表,有一个int类型的12345.我们循环输出. list=[1,2,3,4,5] for i in list: print(i) for i i ...

  4. Python之迭代器和生成器

    Python 迭代器和生成器 迭代器 Python中的迭代器为类序列对象(sequence-like objects)提供了一个类序列的接口,迭代器不仅可以对序列对象(string.list.tupl ...

  5. 【Python】迭代器、生成器、yield单线程异步并发实现详解

    转自http://blog.itpub.net/29018063/viewspace-2079767 大家在学习python开发时可能经常对迭代器.生成器.yield关键字用法有所疑惑,在这篇文章将从 ...

  6. python的迭代器、生成器、装饰器

    迭代器.生成器.装饰器 在这个实验里我们学习迭代器.生成器.装饰器有关知识. 知识点 迭代器 生成器 生成器表达式 闭包 装饰器 实验步骤 1. 迭代器 Python 迭代器(Iterators)对象 ...

  7. Python之迭代器,生成器

    迭代器 1.什么是可迭代对象 字符串.列表.元组.字典.集合都可以被for循环,说明他们都是可迭代的. from collections import Iterable l = [1,2,3,4] t ...

  8. python之迭代器、生成器与面向过程编程

    目录 一 迭代器 二 生成器 三 面向过程编程 一.迭代器 1.迭代器的概念理解 ''' 迭代器从字面上理解就是迭代的工具.而迭代是每次的开始都是基于上一次的结果,不是周而复始的,而是不断发展的. ' ...

  9. day13 python学习 迭代器,生成器

    1.可迭代:当我们打印 print(dir([1,2]))   在出现的结果中可以看到包含 '__iter__', 这个方法,#次协议叫做可迭代协议 包含'__iter__'方法的函数就是可迭代函数 ...

  10. Python之迭代器及生成器

    一. 迭代器 1.1 什么是可迭代对象 字符串.列表.元组.字典.集合 都可以被for循环,说明他们都是可迭代的. 我们怎么来证明这一点呢? from collections import Itera ...

随机推荐

  1. LOJ#6047. 「雅礼集训 2017 Day10」决斗(set)

    题面 传送门 题解 这么简单一道题我考试的时候居然只打了\(40\)分暴力? 如果我们把每个点的\(a_i\)记为\(deg_i-1\),其中\(deg_i\)表示有\(deg_i\)个数的\(A_i ...

  2. 分析解决Java运行时异常

            1 ,基础知识    http://my.oschina.net/u/140462/blog/421128    JVM运行时内存结构          2 ,相关命令    http ...

  3. iOS开发中断言的使用—NSAssert()

    原文链接:http://blog.csdn.net/univcore/article/details/16859263 断言(assertion)是指在开发期间使用的.让程序在运行时进行自检的代码(通 ...

  4. SSAS 收藏

    1.多事实表 SQL实现和SSAS中MDX实现的差异 2.层次结构 3.MDX常用几种查询对比 4.一段话理解 MDX中的Select .轴.COLUMNS.ROWS 5.[转载]MSDN-MDX#0 ...

  5. easyui实现多选框,并且获取值

    在easyui官方文档里面是没有combobox下拉框,可以进行多选的,但是其实是可以多选的, <td align="left">大区:   <input typ ...

  6. 并发编程>>概念准备(一)

    工于其善,必先利器 1.并发和并行的区别 并行:同一时间点执行多个任务(CPU多核或多个CPU同时执行多个任务) 并发:同一时间段内行多个任务(单核同时执行多个任务) 2.同步和异步的区别 同步:执行 ...

  7. Java实现简单井字棋

    Java第一次实验,老师让做一个井字棋,电脑随机下棋. 然后就想能不能聪明一点,可以判断出走哪一步棋:然后只能做到不会输,还是不够聪明,只能呆板地堵住用户,smartRobot的第三个判断逻辑找不到最 ...

  8. <VS2017> 编写VC++单元测试 -(一)新建单元测试工程

    开发人员自己编写单元测试是一个非常好的习惯.单元测试不但能够验证自己所编写的代码是否存在问题,避免提交给测试人员时才发现bug,也可以为将来改动代码的人提供验证代码功能正确性的途径.在我有限的工作生涯 ...

  9. 虚拟机下设置CentOS 7使用固定IP地址

    1.设置虚拟机使用桥接网络 2.查看安装虚拟机软件的电脑IP信息 3.启动CentOS 7进行设置

  10. SpringMVC初写(五)拦截器

    在系统开发过程中,拦截器的使用可以使我们实现一些需求.如:登录认证,权限管理等,拦截器的工作核心就是将一些工作流程进行统一处理 拦截器和过滤器的区别: 过滤器过滤的是请求路径,拦截器拦截的各层方法的映 ...