4

根据关于ABCs的文档,我应该只需要添加一个next方法就可以进行子类化collections.Iterator。所以,我正在使用以下类:

class DummyClass(collections.Iterator):
    def next(self):
        return 1

但是,当我尝试实例化它时出现错误:

>>> x = DummyClass()
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: Can't instantiate abstract class DummyClass with abstract methods __next__

我猜我在做一些愚蠢的事情,但我无法弄清楚它是什么。任何人都可以对此有所了解吗?我可以添加一个__next__方法,但我的印象是只适用于 C 类。

4

2 回答 2

10

看起来您正在使用 Python 3.x。您的代码在 Python 2.x 上运行良好。

>>> import collections
>>> class DummyClass(collections.Iterator):
...     def next(self):
...         return 1
... 
>>> x = DummyClass()
>>> zip(x, [1,2,3,4])
[(1, 1), (1, 2), (1, 3), (1, 4)]

但是在 Python 3.x 上,您应该实现__next__而不是,如py3k docnext的表中所示。(记得阅读正确的版本!)

>>> import collections
>>> class DummyClass(collections.Iterator):
...     def next(self):
...         return 1
... 
>>> x = DummyClass()
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: Can’t instantiate abstract class DummyClass with abstract methods __next__
>>> class DummyClass3k(collections.Iterator):
...     def __next__(self):
...         return 2
... 
>>> y = DummyClass3k()
>>> list(zip(y, [1,2,3,4]))
[(2, 1), (2, 2), (2, 3), (2, 4)]

此更改由PEP-3114 — 重命名iterator.next()iterator.__next__().

于 2010-09-13T19:44:32.393 回答
2

不必从Iterator类继承以使对象可迭代。但在我的情况下,这是必需的,所以它是这样的:

from collections.abc import Iterator

class MyIterator(Iterator):

    def __init__(self, items, property1, property2):
        self.items = items
        self.property1 = property1
        self.property2 = property2
    
    def __iter__(self):
        return self.items.__iter__()

    def __next__(self):
        return self.items.__next__()

测试代码为:

my_iterator = MyIterator([1, 2, 3], "value 1", 2)

for item in my_iterator:
    print(f'{item} ', end='')
print()
print(my_iterator.property1)
print(my_iterator.property2)

输出是:

1 2 3
value 1
2

文档是:collections.abc.Iterator

于 2021-09-22T17:28:56.843 回答