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 Cnameメソッドに戻ります。
kevinarpe 2014年

回答:


117

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):
        ...

1
@James Python 2と互換性のある方法にする方法は?
himanshu219

@James実際に私は両方のためのものが、用事、私はあなたのソリューションに基づいて回答を掲載
himanshu219

45

Python 3.3までは、@abstractmethodおよびをネストすることはできません@property

@abstractproperty抽象プロパティの作成に使用します(docs)。

from abc import ABCMeta, abstractmethod, abstractproperty

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

コードは正しい例外を発生させます。

トレースバック(最新の呼び出しが最後):
  ファイル「foo.py」、36行目、 
    b1 = Base_1( 'abc')  
TypeError:抽象メソッド名で抽象クラスBase_1をインスタンス化できません

42
実際、この答えは若いpythonでは間違っています。3.3以降、@abstractpropertyOPのような組み合わせのために廃止されました。
空飛ぶ羊



3.3までは、ちょうどraise NotImplementedError
S –awomir Lenart

オブジェクトから継承してもABCアノテーションを使用できますか?例のように動作しますか?
santhosh kumar

3

上記のジェームズの回答に基づく

def compatibleabstractproperty(func):

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

それをデコレータとして使用します

@compatibleabstractproperty
def env(self):
    raise NotImplementedError()
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.