72

我已经阅读了关于抽象基类的 python 文档:

这里

abc.abstractmethod(function) 指示抽象方法的装饰器。

使用这个装饰器需要类的元类是ABCMeta或派生自它。ABCMeta 除非重写其所有抽象方法和属性,否则无法实例化具有派生元类的类。

这里

您可以将@abstractmethod装饰器应用于必须实现的方法,例如draw();然后,Python 将为未定义该方法的类引发异常。请注意,仅当您实际尝试创建缺少该方法的子类的实例时才会引发异常。

我已经使用此代码进行了测试:

import abc

class AbstractClass(object):
  __metaclass__ = abc.ABCMeta

  @abc.abstractmethod
  def abstractMethod(self):
    return

class ConcreteClass(AbstractClass):
  def __init__(self):
    self.me = "me"

c = ConcreteClass()
c.abstractMethod()

代码很好,所以我不明白。如果我输入c.abstractMethod,我会得到:

<bound method ConcreteClass.abstractMethod of <__main__.ConcreteClass object at 0x7f694da1c3d0>>

我在这里缺少什么?ConcreteClass 必须实现抽象方法,但我也不例外。

4

2 回答 2

90

您是否使用 python3 来运行该代码?如果是,您应该知道在 python3 中声明元类有更改,您应该这样做:

import abc

class AbstractClass(metaclass=abc.ABCMeta):

  @abc.abstractmethod
  def abstractMethod(self):
      return

答案背后的完整代码和解释是:

import abc

class AbstractClass(metaclass=abc.ABCMeta):

    @abc.abstractmethod
    def abstractMethod(self):
        return

class ConcreteClass(AbstractClass):

    def __init__(self):
        self.me = "me"

# Will get a TypeError without the following two lines:
#   def abstractMethod(self):
#       return 0

c = ConcreteClass()
c.abstractMethod()

如果abstractMethod没有为 定义ConcreteClass,运行上述代码时会引发以下异常:TypeError: Can't instantiate abstract class ConcreteClass with abstract methods abstractMethod

于 2011-08-25T20:02:26.323 回答
13

ABCabc您自己的抽象类导入并使其成为其子类ABC可以帮助使代码看起来更干净。

from abc import ABC, abstractmethod

class AbstractClass(ABC):

  @abstractmethod
  def abstractMethod(self):
    return

class ConcreteClass(AbstractClass):
  def __init__(self):
    self.me = "me"

# The following would raise a TypeError complaining 
# abstractMethod is not implemented
c = ConcreteClass()  

使用 Python 3.6 测试

于 2021-04-27T13:55:25.507 回答