68

在 python 类中,@property 是一个很好的装饰器,它避免使用显式的 setter 和 getter 函数。但是,它的开销是“经典”类函数的 2-5 倍。在我的情况下,这在设置属性的情况下是相当好的,与设置时需要完成的处理相比,开销是微不足道的。

但是,我在获得财产时不需要处理。它始终只是“返回 self.property”。有没有一种优雅的方式来使用 setter 但不使用 getter,而不需要使用不同的内部变量?

只是为了说明,下面的类具有引用内部变量“_var”的属性“var”。调用“var”比调用“_var”需要更长的时间,但如果开发人员和用户都可以只使用“var”而不必跟踪“_var”,那就太好了。

class MyClass(object):
  def __init__(self):
    self._var = None

  # the property "var". First the getter, then the setter
  @property
  def var(self):
    return self._var
  @var.setter
  def var(self, newValue):
    self._var = newValue
    #... and a lot of other stuff here

  # Use "var" a lot! How to avoid the overhead of the getter and not to call self._var!
  def useAttribute(self):
    for i in xrange(100000):
      self.var == 'something'

对于那些感兴趣的人,在我的电脑上调用“var”平均需要 204 ns,而调用“_var”平均需要 44 ns。

4

4 回答 4

82

property在这种情况下不要使用 a 。对象是一个property数据描述符,这意味着任何访问都instance.var将调用该描述符,Python 永远不会在实例本身上查找属性。

您有两个选择:使用.__setattr__()钩子或构建仅实现的描述符.__set__

使用.__setattr__()钩子

class MyClass(object):
    var = 'foo'

    def __setattr__(self, name, value):
        if name == 'var':
            print "Setting var!"
            # do something with `value` here, like you would in a
            # setter.
            value = 'Set to ' + value
        super(MyClass, self).__setattr__(name, value)

现在在读取 .var时使用普通属性查找,但在分配给.var方法时改为__setattr__调用,让您value根据需要拦截和调整它。

演示:

>>> mc = MyClass()
>>> mc.var
'foo'
>>> mc.var = 'bar'
Setting var!
>>> mc.var
'Set to bar'

设置器描述符

setter 描述符只会拦截变量赋值:

class SetterProperty(object):
    def __init__(self, func, doc=None):
        self.func = func
        self.__doc__ = doc if doc is not None else func.__doc__
    def __set__(self, obj, value):
        return self.func(obj, value)

class Foo(object):
    @SetterProperty
    def var(self, value):
        print 'Setting var!'
        self.__dict__['var'] = value

请注意我们需要如何分配给实例.__dict__属性以防止再次调用设置器。

演示:

>>> f = Foo()
>>> f.var = 'spam'
Setting var!
>>> f.var = 'ham'
Setting var!
>>> f.var
'ham'
>>> f.var = 'biggles'
Setting var!
>>> f.var
'biggles'
于 2013-07-10T16:31:26.817 回答
41

propertypython 文档:https ://docs.python.org/2/howto/descriptor.html#properties

class MyClass(object):
    def __init__(self):
        self._var = None

    # only setter
    def var(self, newValue):
        self._var = newValue

    var = property(None, var)


c = MyClass()
c.var = 3
print ('ok')
print (c.var)

输出:

ok
Traceback (most recent call last):
  File "Untitled.py", line 15, in <module>
    print c.var
AttributeError: unreadable attribute
于 2016-01-27T09:41:17.773 回答
3

@WeizhongTu 的回答

class MyClass(object):
    def __init__(self):
        self._var = None

    # only setter
    def var(self, newValue):
        self._var = newValue

    var = property(None, var)


c = MyClass()
c.var = 3
print ('ok')
print (c.var)

很好,除了使变量无法获取的事实......

类似的解决方案但保留吸气剂

var = property(lambda self: self._var, var)

代替

var = property(None, var)
于 2020-05-22T11:29:18.647 回答
2

如果它自己设置属性,则接受的答案的设置器描述符可能会更方便:

一个 setter 描述符 (alt.)

class setter:
    def __init__(self, func, doc=None):
        self.func = func
        self.__doc__ = doc or func.__doc__

    def __set__(self, obj, value):
        obj.__dict__[self.func.__name__] = self.func(obj, value)

class Foo:
    @setter
    def var(self, value):
        print('Setting var!')

        # validations and/or operations on received value
        if not isinstance(value, str):
            raise ValueError('`var` must be a string')
        value = value.capitalize()

        # returns property value
        return value

演示:

>>> f = Foo()
>>> f.var = 'spam'
Setting var!
>>> f.var = 'ham'
Setting var!
>>> f.var
'Ham'
>>> f.var = 'biggles'
Setting var!
>>> f.var
'Biggles'
>>> f.var = 3
ValueError: `var` must be a string
于 2020-02-25T21:10:56.130 回答