5

假设您在基类中有一个属性,该属性具有setter将在所有子类中使用的单个方法,但在所有子类中具有不同getter的方法。理想情况下,您只想setter在基类中为该方法编写一次代码。此外,您想让基类抽象,因为它没有完整的getter方法。有没有办法做到这一点?

天真地,我认为它应该是这样的:

class _mystring(object):

    __metaclass__ = abc.ABCMeta

    @abc.abstractproperty
    def str(self):
        pass

    @str.setter
    def str(self,value):
        self._str = value

class uppercase(_mystring):

    @_mystring.str.getter
    def str(self):
        return self._str.upper()

但是upper = uppercase()失败了Can't instantiate abstract class uppercase with abstract methods str

如果我将@abc.abstractpropertyin class更改_mystring@property, 那么upper = uppercase()可以,但是upper = _mystring(), 我不想要的。

我错过了什么?

谢谢!

4

1 回答 1

3

我设法制作了一个小包装器来abstractproperty做到这一点:

class partialAP(abc.abstractproperty):
    def getter(self, func):
        if getattr(func, '__isabstractmethod__', False) or getattr(self.fset, '__isabstractmethod__', False):
            p = partialAP(func, self.fset)
        else:
            p = property(func, self.fset)
        return p

    def setter(self, func):
        if getattr(self.fget, '__isabstractmethod__', False) or getattr(func, '__isabstractmethod__', False):
            p = partialAP(self.fget, func)
        else:
            p = property(self.fset, func)
        return p

然后您的代码只需稍作修改即可工作:

class _mystring(object):

    __metaclass__ = abc.ABCMeta

    @partialAP
    @abc.abstractmethod
    def str(self):
        pass

    @str.setter
    def str(self,value):
        self._str = value

class uppercase(_mystring):

    @_mystring.str.getter
    def str(self):
        return self._str.upper()

然后行为是所希望的:

>>> _mystring()
Traceback (most recent call last):
  File "<pyshell#3>", line 1, in <module>
    _mystring()
TypeError: Can't instantiate abstract class _mystring with abstract methods str
>>> u = uppercase()
>>> u.str = 'a'
>>> u.str
'A'

我的partialAP包装器必须与abc.abstractmethod. 您所做的是abstractmethod用来装饰您想要抽象的属性(getter 或 setter)的“部分”,并partialAP用作第二个装饰器。 partialAP定义 getter/setter 函数,仅当 getter 和 setter 都是非抽象的时,才用普通属性替换属性。

显然,您必须对其进行一些扩展,以使其也值得使用属性删除器。如果您有更复杂的继承层次结构,也可能存在无法正确处理的极端情况。

后代的旧解决方案:

class _mystring(object):

    __metaclass__ = abc.ABCMeta

    @abc.abstractmethod
    def _strGet(self):
        pass

    def _strSet(self,value):
        self._str = value

    str = property(_strGet, _strSet)

class uppercase(_mystring):

    def _strGet(self):
        return self._str.upper()
    str = _mystring.str.getter(_strGet)

然后它工作:

>>> _mystring()
Traceback (most recent call last):
  File "<pyshell#39>", line 1, in <module>
    _mystring()
TypeError: Can't instantiate abstract class _mystring with abstract methods _strGet
>>> u = uppercase()
>>> u.str = 'a'
>>> u.str
'A'

诀窍是您不必使用方便的装饰器。装饰器将abstractproperty整个属性标记为抽象。您要做的是创建两个方法,一个抽象的(用于 getter)和一个常规的(用于 setter),然后创建一个组合它们的常规属性。然后,当您扩展类时,您必须重写抽象 getter 并将其与基类属性显式“混合”(使用_mystring.str.getter)。

您不能使用的原因@_mystring.str.getter是装饰器强制 getter 和 setter 与属性本身具有相同的名称。如果您只想将两者中的一个标记为摘要,则它们必须具有不同的名称,否则您无法分别获得它们。

请注意,此解决方案要求子类为它们的 getter 提供正确的名称(_strGet在这种情况下),以便满足它们已覆盖抽象方法的 ABC。

于 2013-10-31T21:24:05.593 回答