我已经阅读了相关资料,其中说明Python迭代器必须同时具有__iter__
和__next__
方法,但可迭代对象仅需要__iter__
方法。
我检查了一个列表,并发现它没有__next__
方法。当使用iter()
在其上进行操作时,它将变成一个迭代器。这是否意味着iter()
会向列表中添加一个__next__
方法以将其转换为迭代器?
如果是,这是如何发生的?
我已经阅读了相关资料,其中说明Python迭代器必须同时具有__iter__
和__next__
方法,但可迭代对象仅需要__iter__
方法。
我检查了一个列表,并发现它没有__next__
方法。当使用iter()
在其上进行操作时,它将变成一个迭代器。这是否意味着iter()
会向列表中添加一个__next__
方法以将其转换为迭代器?
如果是,这是如何发生的?
iter
不会将列表转换为迭代器,它返回一个迭代器。它不会修改列表,并且列表当然没有 __next__
方法。
>>> x = [1,2]
>>> it = iter(x)
>>> it
<list_iterator object at 0x101c021d0>
>>> x.__next__
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
AttributeError: 'list' object has no attribute '__next__'
>>>
列表是可迭代对象(iterables),而不是迭代器(iterators)。它们实现了__iter__
方法,因此它们是可迭代的:
>>> x.__iter__
<method-wrapper '__iter__' of list object at 0x101bcf248>
__next__
,因此它们不是迭代器:>>> next(x)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: 'list' object is not an iterator
按照定义,迭代器本身是可迭代的,因为它们也实现了__iter__
。考虑以下代码:
>>> x = [1, 2, 3, 4, 5, 6, 7, 8, 9]
>>> it = iter(x)
>>> it
<list_iterator object at 0x101c02358>
>>> it.__iter__
<method-wrapper '__iter__' of list_iterator object at 0x101c02358>
大多数迭代器在使用iter
时应该简单地返回自身:
>>> it2 = iter(it)
>>> it, it2
(<list_iterator object at 0x101c02358>, <list_iterator object at 0x101c02358>)
>>> it is it2
True
>>>
确实,这是迭代器协议的一个要求:
“迭代器需要有一个返回迭代器对象本身的
__iter__()
方法,所以每个迭代器也都是可迭代的,并且可以在大多数接受其他可迭代对象的地方使用。”
请注意,它们是相同的迭代器:
>>> next(it)
1
>>> next(it2)
2
>>> next(it)
3
>>> next(it)
4
>>> next(it2)
5
>>> list(it)
[6, 7, 8, 9]
>>> next(it2)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
StopIteration
__iter__
和__next__
,可迭代对象只是指它实现了__iter__
。由__iter__
返回的是一个迭代器,因此必须实现__next__
。__next__
方法在 iter
返回的对象上。 - Eevee__iter__
实现,该实现返回迭代器对象本身,而不仅仅是“大多数”。这是迭代器协议的要求:“迭代器需要有一个返回迭代器对象本身的__iter __()方法,因此每个迭代器也是可迭代的,并且可以在其他可迭代接受的大多数地方使用。” Python的其他部分依赖于此; 一种规范和高效的鸭子类型方法来测试某些东西是否为迭代器是测试obj is iter(someobj)
。 - ShadowRanger