43

我正在研究一种库,出于一个奇怪的原因,我遇到了这个错误。

  • 是我的代码。当然@abc.abstractmethod 必须取消注释
  • 是我的测试

抱歉不能直接复制粘贴

我的基础是下面的代码有效

测试.py

import abc
import six

@six.add_metaclass(abc.ABCMeta)
class Base(object):

    @abc.abstractmethod
    def whatever(self,):
        raise NotImplementedError

class SubClass(Base):

    def __init__(self,):

        super(Base, self).__init__()
        self.whatever()

    def whatever(self,):
        print("whatever")

在 python 外壳中

>>> from test import *
>>> s = SubClass()
whatever

为什么我的名册模块出现此错误

Can't instantiate abstract class Player with abstract methods _Base__json_builder, _Base__xml_builder

提前致谢

4

1 回答 1

58

您的问题出现是因为您已经在基础抽象类中定义了带有__(双下划线)前缀的抽象方法。这会导致 python在定义类时进行名称修改。

函数的名称从__json_builderto_Base__json_builder__xml_builderto更改_Base__xml_builder。这是您必须在子类中实现/覆盖的名称。

要在您的示例中显示此行为 -

>>> import abc
>>> import six
>>> @six.add_metaclass(abc.ABCMeta)
... class Base(object):
...     @abc.abstractmethod
...     def __whatever(self):
...             raise NotImplementedError
...
>>> class SubClass(Base):
...     def __init__(self):
...             super(Base, self).__init__()
...             self.__whatever()
...     def __whatever(self):
...             print("whatever")
...
>>> a = SubClass()
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: Can't instantiate abstract class SubClass with abstract methods _Base__whatever

当我将实现更改为以下内容时,它可以工作

>>> class SubClass(Base):
...     def __init__(self):
...             super(Base, self).__init__()
...             self._Base__whatever()
...     def _Base__whatever(self):
...             print("whatever")
...
>>> a = SubClass()
whatever

但这很乏味,您可能要考虑是否真的想用__(双下划线) 定义函数。您可以在此处阅读有关名称修改的更多信息。

于 2015-07-16T15:32:48.453 回答