19

私のクラスの1つには、取得と設定で非常によく似た処理を行うプロパティがいくつかあります。そこで、引数をpropertyファクトリ関数に抽象化しました。

def property_args(name):
    def getter(self):
        # do something
        return getattr(self, '_' + name)
    def setter(self, value)
        # do something
        setattr(self, '_' + name, value)
    return getter, setter

class MyClass(object):
    def __init__(self):
        self._x = None
    x = property(*property_args('x'))  # obviously there's more than one of these IRL

しかし、それ以来、それpropertyが実際にはクラスであり、サブクラス化することがこれに最適であることを発見しました。オーバーライドする必要があるもの(およびの引数シグネチャなど)を説明するドキュメントには何も見つかり__init__ません。また、Cソースコードを探しに行く必要はありません。誰かが私がこの情報を見つけることができる場所を知っていますか?

4

2 に答える 2

26

これは、 property()のコードに相当する純粋なPythonです。

class Property(object):
    "Emulate PyProperty_Type() in Objects/descrobject.c"

    def __init__(self, fget=None, fset=None, fdel=None, doc=None):
        self.fget = fget
        self.fset = fset
        self.fdel = fdel
        if doc is None and fget is not None:
            doc = fget.__doc__
        self.__doc__ = doc

    def __get__(self, obj, objtype=None):
        if obj is None:
            return self
        if self.fget is None:
            raise AttributeError("unreadable attribute")
        return self.fget(obj)

    def __set__(self, obj, value):
        if self.fset is None:
            raise AttributeError("can't set attribute")
        self.fset(obj, value)

    def __delete__(self, obj):
        if self.fdel is None:
            raise AttributeError("can't delete attribute")
        self.fdel(obj)

    def getter(self, fget):
        return type(self)(fget, self.fset, self.fdel, self.__doc__)

    def setter(self, fset):
        return type(self)(self.fget, fset, self.fdel, self.__doc__)

    def deleter(self, fdel):
        return type(self)(self.fget, self.fset, fdel, self.__doc__)
于 2012-09-13T11:50:20.293 に答える
7

property記述子であるため、、、、およびメソッドをオーバーライド(または新しいオブジェクトで再実装)する必要が__get__()あり__set__()__delete__()ます。

于 2012-09-13T11:50:24.697 に答える