是否可以在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
包这样的东西是否仍然可以工作。
有一些标准的方法可以处理吗?
答案 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
实例变量。 (我不知道静态类型检查器是否会理解该实例变量应该来自子类。)