gpt4 book ai didi

Python 类@property : use setter but evade getter?

转载 作者:IT老高 更新时间:2023-10-28 20:34:08 27 4
gpt4 key购买 nike

在 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。

最佳答案

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

您有两个选择:使用 .__setattr__() Hook 或构建仅实现 .__set__.

的描述符

使用 .__setattr__() 钩子(Hook)

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 描述符

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'

关于Python 类@property : use setter but evade getter?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17576009/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com