Python的子类化`property`

20

在我的一个类中,我有一些在获取和设置时执行非常相似操作的属性。因此,我将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 源代码中寻找答案。有没有人知道我可以在哪里找到这些信息呢?


1
参考文献,C语言实现可以在此处找到:http://hg.python.org/cpython/file/tip/Objects/descrobject.c - nneonneo
@nneonneo:谢谢!1228行的大型注释块就是答案。如果你想把它写成完整的答案,那我会接受它。 - Benjamin Hodgson
可怜的家伙,你可以回答自己的问题 :) - Burhan Khalid
1
@BurhanKhalid 但是nneonneo应该得到功劳! - Benjamin Hodgson
2个回答

28

以下是与 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__)

1
这现在已经在Python 3的在线文档中,包含在描述符指南中(由Raymond编写)。 (它也在Python 2文档中。) - martineau

7

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接