__iter__,__next__
__iter__将一个对象变为可迭代对象。
__next__将一个对象变为迭代器。
⚠️迭代器一定是可迭代的,所以一个对象如想成为一个迭代器,必须既要实现__iter__方法,也要实现__next__方法
iter(xxx)方法会自动调用xxx.__iter__方法
next(xxx)方法会自动调用xxx.__next__方法
看下面代码:
from collections.abc import Iterable
from collections.abc import Iterator
class Classmate(object):
def __init__(self):
self.names=list();
def add(self,name):
self.names.append(name)
classmate = Classmate()
classmate.add("张三")
classmate.add("李四")
classmate.add("王五")
print(isinstance(classmate,Iterable)) # 不是可迭代对象,输出 False
print(isinstance(classmate,Iterator)) # 不是迭代器,输出 False
for name in classmate:
print(name)
# 报错
"""
Traceback (most recent call last):
File "/Users/xxx/PycharmProjects/untitled/xxxx.py", line 20, in <module>
for name in classmate:
TypeError: 'Classmate' object is not iterable
"""
代码中想要通过for循环,遍历打印classmate对象中的names列表,报错TypeError: 'Classmate' object is not iterable,因为classmate对象不是一个可迭代对象。那如何将其变成可迭代对象呢?
如下修改代码:
from collections.abc import Iterable
from collections.abc import Iterator
class Classmate(object):
def __init__(self):
self.names=list();
def add(self,name):
self.names.append(name)
def __iter__(self):
"""如果想要一个对象成为一个可迭代对象,既可以使用for,那么必须实现__iter__方法"""
pass
classmate = Classmate()
classmate.add("张三")
classmate.add("李四")
classmate.add("王五")
print(isinstance(classmate,Iterable)) # 是可迭代对象,输出 True
print(isinstance(classmate,Iterator)) # 不是迭代器,输出 False
for name in classmate:
print(name)
# 报错
"""
Traceback (most recent call last):
File "/Users/xxx/PycharmProjects/untitled/xxxxx.py", line 23, in <module>
for name in classmate:
TypeError: iter() returned non-iterator of type 'NoneType'
"""
可以看到,修改后
print(isinstance(classmate,Iterable))
输出 True,说明classmate已经是可迭代对象了。但是依然报错了TypeError: iter() returned non-iterator of type 'NoneType',这是因为classmate对象的__iter__必须要返回一个拥有__iter__和__next__方法的对象的引用。
再次修改代码:
from collections.abc import Iterable
from collections.abc import Iterator
class Classmate(object):
def __init__(self):
self.names=list();
def add(self,name):
self.names.append(name)
def __iter__(self):
"""如果想要一个对象成为一个可迭代对象,既可以使用for,那么必须实现__iter__方法"""
return ClassmateIterator()
class ClassmateIterator(object):
def __iter__(self):
pass
def __next__(self):
pass
classmate = Classmate()
classmate.add("张三")
classmate.add("李四")
classmate.add("王五")
print(isinstance(classmate,Iterable)) # 是可迭代对象,输出 True
print(isinstance(classmate,Iterator)) # 不是迭代器,输出 False
for name in classmate:
print(name)
运行程序,发现可以正常运行,控制台一直输出None,这是因为没有设置迭代器返回的条件。
继续修改代码,如下:
from collections.abc import Iterable
from collections.abc import Iterator
class Classmate(object):
def __init__(self):
self.names=list();
def add(self,name):
self.names.append(name)
def __iter__(self):
"""如果想要一个对象成为一个可迭代对象,既可以使用for,那么必须实现__iter__方法"""
return ClassmateIterator(self)
class ClassmateIterator(object):
def __init__(self,obj):
self.obj = obj
self.index = 0
def __iter__(self):
pass
def __next__(self):
if self.index<len(self.obj.names):
ret = self.obj.names[self.index]
self.index += 1
return ret
else:
raise StopIteration
classmate = Classmate()
classmate.add("张三")
classmate.add("李四")
classmate.add("王五")
print(isinstance(classmate,Iterable)) # 是可迭代对象,输出 True
print(isinstance(classmate,Iterator)) # 不是迭代器,输出 False
for name in classmate:
print(name)
# 张三
# 李四
# 王五
扩展
可能有的人,会说可不可以把类Classmate 和 类ClassmateIterator合并呢?如下:
from collections.abc import Iterable
from collections.abc import Iterator
class Classmate(object):
def __init__(self):
self.names=list()
self.index = 0
def add(self,name):
self.names.append(name)
def __iter__(self):
"""如果想要一个对象成为一个可迭代对象,既可以使用for,那么必须实现__iter__方法"""
return self
def __next__(self):
if self.index<len(self.names):
ret = self.names[self.index]
self.index += 1
return ret
else:
raise StopIteration
classmate = Classmate()
classmate.add("张三")
classmate.add("李四")
classmate.add("王五")
print(isinstance(classmate,Iterable)) # 是可迭代对象,输出 True
print(isinstance(classmate,Iterator)) # 是迭代器,输出 True
for name in classmate:
print(name)
# 张三
# 李四
# 王五
⚠️ 这样看起来并没有什么错误,输出的结果也是对的,但是如果想打印如下结果:
张三 张三
张三 李四
张三 王五
李四 张三
李四 李四
李四 王五
王五 张三
王五 李四
王五 王五
即:
for name in classmate:
for name1 in classmate:
print(name,name1)
# 张三 李四
# 张三 王五
发现输出的结果并不是我们预期的结果,这是因为俩个for循环所引用的迭代器是同一个所导致的(self.index在俩个for循环使用的是同一引用)
解决方案
那该如何解决呢???
- 将类Classmate 和 类ClassmateIterator分开,即上文的代码。
from collections.abc import Iterable
from collections.abc import Iterator
class Classmate(object):
def __init__(self):
self.names=list();
def add(self,name):
self.names.append(name)
def __iter__(self):
"""如果想要一个对象成为一个可迭代对象,既可以使用for,那么必须实现__iter__方法"""
return ClassmateIterator(self)
class ClassmateIterator(object):
def __init__(self,obj):
self.obj = obj
self.index = 0
def __iter__(self):
pass
def __next__(self):
if self.index<len(self.obj.names):
ret = self.obj.names[self.index]
self.index += 1
return ret
else:
raise StopIteration
classmate = Classmate()
classmate.add("张三")
classmate.add("李四")
classmate.add("王五")
print(isinstance(classmate,Iterable)) # 是可迭代对象,输出 True
print(isinstance(classmate,Iterator)) # 不是迭代器,输出 False
for name in classmate:
for name1 in classmate:
print(name,name1)
- 使用self.names本身自己的迭代器返回
from collections.abc import Iterable
from collections.abc import Iterator
class Classmate(object):
def __init__(self):
self.names=list()
self.index = 0
def add(self,name):
self.names.append(name)
def __iter__(self):
ii = iter(self.names)
return ii
classmate = Classmate()
classmate.add("张三")
classmate.add("李四")
classmate.add("王五")
print(isinstance(classmate,Iterable)) # 是可迭代对象,输出 True
print(isinstance(classmate,Iterator)) # 不是迭代器,输出 False
for name in classmate:
for name1 in classmate:
print(name,name1)
- 利用yield生成器
from collections.abc import Iterable
from collections.abc import Iterator
class Classmate(object):
def __init__(self):
self.names=list()
self.index = 0
def add(self,name):
self.names.append(name)
def __iter__(self):
for v in self.names:
yield v
classmate = Classmate()
classmate.add("张三")
classmate.add("李四")
classmate.add("王五")
print(isinstance(classmate,Iterable)) # 是可迭代对象,输出 True
print(isinstance(classmate,Iterator)) # 不是迭代器,输出 False
for name in classmate:
for name1 in classmate:
print(name,name1)
网友评论