Python类@property:使用setter但可以逃避getter?

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

但是,获取属性时不需要任何处理。 它始终只是“返回自我属性”。 是否有一种优雅的方式来使用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。

Jonas Lindeløv asked 2020-02-13T18:38:37Z
2个解决方案
59 votes

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

您有两个选择:使用.var钩子或构建仅实现.var的描述符。

使用.var挂钩

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时,将调用.__dict__方法,从而拦截value并根据需要进行调整。

演示:

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

setter描述符

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__属性,以防止再次调用setter。

演示:

>>> f = Foo()
>>> f.var = 'spam'
Setting var!
>>> f.var = 'ham'
Setting var!
>>> f.var
'ham'
>>> f.var = 'biggles'
Setting var!
>>> f.var
'biggles'
Martijn Pieters answered 2020-02-13T18:39:27Z
28 votes

property python文档:[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
WeizhongTu answered 2020-02-13T18:39:51Z
translate from https://stackoverflow.com:/questions/17576009/python-class-property-use-setter-but-evade-getter