Python:抽象实例变量?

时间:2018-06-27 06:17:16

标签: python abstract-class python-3.6 instance-variables

是否可以在python中为类声明抽象实例变量?

例如,我们有一个抽象基类Bird,其中有一个使用fly包实现的抽象方法abc和抽象实例变量feathers(我正在寻找)实现为属性。

from abc import ABCMeta, abstractmethod

class Bird(metaclass=ABCMeta):

    @property
    @abstractmethod
    def feathers(self):
        """The bird's feathers."""

    @abstractmethod
    def fly(self):
        """Take flight."""

问题在于,Eagle是从Bird派生的类,需要将feathers实现为属性方法。因此,以下类别不是可接受的类别,但我希望它成为

class Eagle(Bird):

    def __init__(self):
        self.feathers = 'quill'

    def fly(self):
        print('boy are my arms tired')

可能是有问题的,因为要求是在实例本身上,并且实际上是在实例化之后,因此我不知道像abc包这样的东西是否仍然可以工作。

有一些标准的方法可以处理吗?

2 个答案:

答案 0 :(得分:2)

使用通用属性,可以像下面这样简单地工作:

class Bird(object):
    @property
    def feathers(self):
        try:
            return self._feathers
        except AttributeError:
            raise WhateverError('No feathers')  # maybe obfuscate inner structure

class Eagle(Bird):
    def __init__(self):
        self._feathers = 'quill'

>>> Bird().feathers
WhateverError: No feathers
>>> Eagle().feathers
'quill'

答案 1 :(得分:2)

abc系统不包含声明抽象实例变量的方法。确定类是具体的还是抽象的代码必须在任何实例存在之前运行;它可以很容易地检查类的方法和属性,但是无法判断实例是否将具有特定的实例属性。

最接近的可能是变量注释:

class Bird(metaclass=ABCMeta):
    feathers : str
    ...

abc对该注释不做任何事情,但它至少向读者和静态类型检查器表示Bird的实例应该具有一个feathers实例变量。 (我不知道静态类型检查器是否会理解该实例变量应该来自子类。)