Як створити абстрактні властивості в абстрактних класах python


118

У наступному коді я створюю базовий абстрактний клас Base. Я хочу, щоб усі класи, які успадковуються, Baseнадавали nameмайно, тому я зробив цю властивість @abstractmethod.

Тоді я створив підклас Base, який називається Base_1, який має на меті забезпечити деяку функціональність, але все ще залишається абстрактним. В ньому немає nameвластивості Base_1, але, тим не менш, 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')  

Зрозумів: Якщо ви забули використовувати декоратор @propertyв class C, nameповертається до методу.
kevinarpe

Відповіді:


117

Оскільки в Python 3.3 була виправлена ​​помилка, що означає, що property()декоратор тепер правильно ідентифікується як абстрактний при застосуванні до абстрактного методу.

Примітка. Замовлення має значення, яке ви повинні використовувати @propertyраніше@abstractmethod

Python 3.3+: ( Документи python ):

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

Python 2: ( Документи python )

class C(ABC):
    @abstractproperty
    def my_abstract_property(self):
        ...

1
@James Як зробити його сумісним і для python 2?
himanshu219

@James насправді я мав на увазі і те, і інше, але ніколи не знаю, я опублікував відповідь на основі вашого рішення
himanshu219

45

До Python 3.3 ви не можете гніздо @abstractmethodта @property.

Використовуйте @abstractpropertyдля створення абстрактних властивостей ( документів ).

from abc import ABCMeta, abstractmethod, abstractproperty

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

Код наражає правильний виняток:

Traceback (останній останній дзвінок):
  Файл "foo.py", рядок 36, в 
    b1 = Base_1 ('abc')  
TypeError: Неможливо створити абстрактний клас Base_1 з назвою абстрактних методів

42
насправді ця відповідь неправильна для молодших пітонів: починаючи з 3.3, @abstractpropertyце застаріло на користь такої комбінації, як ОП.
літаючі вівці



тож до 3.3, простоraise NotImplementedError
Славомір Ленарт

чи можемо ми успадкувати об’єкт і все-таки використовувати анотації ABC? Чи буде це працювати, як показано в прикладі?
сантош кумар

3

Виходячи з відповіді Джеймса вище

def compatibleabstractproperty(func):

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

і використовувати його як декоратор

@compatibleabstractproperty
def env(self):
    raise NotImplementedError()
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.