Как создать абстрактные свойства в абстрактных классах Python - PullRequest
68 голосов
/ 11 мая 2011

В следующем коде я создаю базовый абстрактный класс Base.Я хочу, чтобы все классы, которые наследуются от Base, обеспечивали свойство name, поэтому я сделал это свойство @abstractmethod.

Затем я создал подкласс Base, который называется Base_1,который предназначен для предоставления некоторой функциональности, но все еще остается абстрактным.В Base_1 нет свойства name, но, тем не менее, python создает объект этого класса без ошибок.Как создать абстрактные свойства?

from abc import ABCMeta, abstractmethod
class Base(object):
    __metaclass__ = ABCMeta
    def __init__(self, strDirConfig):
        self.strDirConfig = strDirConfig

    @abstractmethod
    def _doStuff(self, signals):
        pass

    @property    
    @abstractmethod
    def name(self):
        #this property will be supplied by the inheriting classes
        #individually
        pass


class Base_1(Base):
    __metaclass__ = ABCMeta
    # this class does not provide the name property, should raise an error
    def __init__(self, strDirConfig):
        super(Base_1, self).__init__(strDirConfig)

    def _doStuff(self, signals):
        print 'Base_1 does stuff'


class C(Base_1):
    @property
    def name(self):
        return 'class C'


if __name__ == '__main__':
    b1 = Base_1('abc')  

Ответы [ 3 ]

49 голосов
/ 09 февраля 2018

Начиная с Python 3.3 исправлена ​​ошибка, означающая, что декоратор property() теперь правильно идентифицируется как абстрактный при применении к абстрактному методу.

Примечание: порядок важен, вы должны использовать@property до @abstractmethod

Python 3.3 +: ( Python Docs ):

class C(ABC):
    @property
    @abstractmethod
    def my_abstract_property(self):
        ...

Python 2: ( Python Docs )

class C(ABC):
    @abstractproperty
    def my_abstract_property(self):
        ...
41 голосов
/ 11 мая 2011

До Python 3.3 , вы не можете вкладывать @abstractmethod и @property.

Используйте @abstractproperty для создания абстрактных свойств ( документы ).

from abc import ABCMeta, abstractmethod, abstractproperty

class Base(object):
    # ...
    @abstractproperty
    def name(self):
        pass

Код теперь вызывает правильное исключение:

Traceback (most recent call last):
  File "foo.py", line 36, in 
    b1 = Base_1('abc')  
TypeError: Can't instantiate abstract class Base_1 with abstract methods name
0 голосов
/ 12 июля 2019

На основании ответа Джеймса выше

def compatibleabstractproperty(func):

    if sys.version_info > (3, 3):             
        return property(abstractmethod(func))
    else:
        return abstractproperty(func)

и использовать его как декоратор

@compatibleabstractproperty
def env(self):
    raise NotImplementedError()
...