python 迭代器 一个奇怪的解决方法
一般我们在类里面写迭代器都是如下写法:
class IterableSomthing:
def __iter__(self):
return self def __next__(self):
return 1
但是,《流畅的python》给出了不同的见解。该书指出,在数据结构内实现迭代器是个很蠢的想法,因为需要引入游标指针记录位置的缘故,这么实现迭代器会造成数据结构空间性能下降,同时,因为游标指针的独立性使得改数据结构无法并发遍历,所以又造成了时间性能的下降。代码如下
class Node:
def __init__(self, item):
self.item = item
self.pre = None
self.next = None class Deque:
def __init__(self):
"""创建一个空的双端队列"""
self.head = None
self.tail = None
self.point = self.head #游标指针 def add_front(self, item):
"""从队头加入一个item元素"""
n0 = Node(item)
if self.head:
n0.next = self.head
self.head.pre = n0
self.head = n0
else:
self.head = n0
self.tail = n0
self.zero() def add_rear(self, item):
"""从队尾加入一个item元素"""
n0 = Node(item)
if self.tail:
n0.pre = self.tail
self.tail.next = n0
self.tail = n0
else:
self.head = n0
self.tail = n0
self.zero() def remove_front(self):
"""从队头删除一个item元素"""
if self.head:
if self.head == self.tail:
self.head = None
self.tail = None
else:
self.head.next.pre = None
self.head = self.head.next
self.zero() def remove_rear(self):
"""从队尾删除一个item元素"""
if self.tail:
if self.head == self.tail:
self.head = None
self.tail = None
else:
self.tail.pre.next = None
self.tail = self.tail.pre
self.zero() def is_empty(self):
"""判断双端队列是否为空"""
return self.head is None def size(self):
"""返回队列的大小"""
i = 0
n0 = self.head
while n0:
i += 1
n0 = n0.next
return i def tolist(self):
li = []
n0 = self.head
while n0:
li.append(n0.item)
n0 = n0.next
return li def gen(self):
n0 = self.head
while n0:
# print(id(self))
yield n0.item
n0 = n0.next
raise StopIteration def __iter__(self):
# n0 = self.head
# while n0:
# # print(id(self))
# yield n0.item
# n0 = n0.next
# raise StopIteration
return self def __next__(self):
if self.point:
n0 = self.point
self.point = self.point.next
return n0.item
else:
self.zero()
raise StopIteration def zero(self): #游标指针归零函数
self.point = self.head
这是个双端队列的python实现,如果实现了迭代器,就必须实现游标指针类属性,游标指针归零类方法,着实降低了开发效率
有没有别的解决方法呢?
有
只要改一处:
def __iter__(self):
__index_temp = self.__head
while __index_temp:
n0 = __index_temp
__index_temp = __index_temp.next
yield n0
else:
raise StopIteration
对你没看错,没有__next__函数!没有游标指针!没有归零函数!
这个__iter__函数需要返回一个迭代器,我们就给他一个,因为生成器也是迭代器!
此时,这个双端队列的python实现就不是一个迭代器了,而是一个可迭代对象,就可以用for循环迭代了
def test2():
s1 = DoubleLinkList()
for i in range(1000):
s1.append(i)
for ii in s1:
print(ii.item)
if ii.item == 500:
print('------------------------------------------------------------------------------')
break
for ii in s1:
print(ii.item)
而且每一个for循环都是独立的,因为每个for循环块获得的对象都是一个独立的生成器,相互之间不会干扰,虽然看起来都是调用同一个对象,但实际上完全不是这么一回事,这就和list的实现是一样一样的,这样的话做并发循环就容易多了。
思考:python里的魔术方法都很有用,但是我们是要为了实现某个功能专门实现对应的魔术方法呢,还是直接实现对应的功能函数呢?
魔术方法是为了其他方法服务的基本方法,还是锦上添花的增补手段呢?
python 迭代器 一个奇怪的解决方法的更多相关文章
- python cmd 窗口 中文乱码 解决方法 (附:打印不同颜色)
python cmd 窗口 中文乱码 解决方法 (附:打印不同颜色) 前言 在 python 开发中,有时候想通过cmd窗口来和用户交互,比如显示信息之类的,会比自己创建 GUI 来的方便,但是随之而 ...
- python __file__ is not defined 解决方法
python __file__ is not defined 解决方法 __file__ 是在python module 被导入的时候生成的一个变量,所以在 __file__ 不能被使用,但是又想获取 ...
- Python更新pip出现错误解决方法
Python更新pip出现错误解决方法 更新pip python -m pip install --upgrade pip 查看时报错 解决方法 在命令栏(即win+r)输入:easy_install ...
- Emgu CV的一个异常的解决方法
今年组里有大项目落我头上了,并不能像去年一样回家还能搞搞Cocos2dX,一把老泪流了下来... 回到正题,由于组里需要做一个显示板的自动测试项目,涉及到Computer Vision.不得不说,这才 ...
- python socket.error: [Errno 10054] 解决方法
我用的是python2.7 我搜网上10054错误解决方法的时候发现,大部分文章都是以python3为基础的,对于python2不适用. python socket.error: [Errno 1 ...
- 一个简单的解决方法:word文档打不开,错误提示mso.dll模块错误。
最近电脑Word无故出现故障,无法打开,提示错误信息如下: 问题事件名称: APPCRASH应用程序名: WINWORD.EXE应用程序版本: 11.0.8328.0应用程序时间戳: 4c717ed1 ...
- C# 该行已经属于另一个表 的解决方法[转]
该文转自:http://blog.sina.com.cn/s/blog_48e4c3fe0100nzs6.html DataTable dt = new DataTable(); dt = ds.Ta ...
- The “SignFile” task was not given a value for the required parameter “CertificateThumbprint”的一个简单的解决方法
这个只是其中一种解决方法,而且不是万能的 1. 由提示内容可以看出,这个一个 sign(认证)的问题, 在出现这个问题的项目上,鼠标右键,选择properties,然后选择signing. 2. 选择 ...
- python 进程内存增长问题, 解决方法和工具
转载:http://drmingdrmer.github.io/tech/programming/2017/05/06/python-mem.html#pyrasite-%E8%BF%9E%E6%8E ...
随机推荐
- http中的socket是怎么一回事
首先我们先看一下socket的定义:是在传输层和应用层中间的一个抽象层,是实现网络通信的接口. 那么什么是传输层,什么是应用层呢?网络又是怎样通信的呢?为了弄清这两个问题,我们需要弄清一个概念TCP/ ...
- DBUtils——handler
ArrayHandler: 把结果集中的第一行数据转成对象数组. ArrayListHandler: 把结果集中的每一行数据都转成一个对象数组,再存放到List中. BeanHandler: 将结果集 ...
- 解决Python图片处理模块pillow使用中出现的问题
最近爬一个电影票房的网站(url:http://58921.com/alltime),上面总票房里面其实是一张图片,那么我需要把图片识别成文字,来获取票房数据. 我头脑里第一想到的解决方案就是要用 ...
- select、poll、epoll的区别
本文写于2017-02-26,从老账号迁移到本账号,原文地址:https://www.cnblogs.com/huangweiyang/p/6444746.html select.poll.epoll ...
- Java泛型之自限定类型
在<Java编程思想>中关于泛型的讲解中,提到了自限定类型: class SelfBounded<T extends SelfBounded<T>> 作者说道: 这 ...
- Java NStruct
package org.rx.bean; import org.rx.Lazy; import org.rx.SystemException; import java.io.Serializable; ...
- asp类型转换函数汇总 转贴
abs(number) 返回绝对值. array(arglist) 创建一个数组. asc(string) 返回字符串第一个字符的ansi码. atn(number) 返回反正弦值. cbool (e ...
- 详细集成Redis (一)
1.添加依赖 <dependency> <groupId>redis.clients</groupId> <artifactId>jedis</a ...
- TCP学习总结(三)
TCP的流量控制 1.利用滑动窗口实现流量控制 所谓流量控制,就是让发送方的发送速率不要太快,要让接收方来得及接收.利用下图来说明如何利用滑动窗口机制进行流量控制: 一开始(connect后),B告诉 ...
- vue-router 重难点总结笔记
1,使用动态路由配置的(如:‘:id’),可以在this.$router.params.id获得. 官网例子: 模式 匹配路径 $route.params /user/:username /user/ ...