我正在尝试使用新式属性声明:
class C(object):
def __init__(self):
self._x = 0
@property
def x(self):
print 'getting'
return self._x
@x.setter
def set_x(self, value):
print 'setting'
self._x = value
if __name__ == '__main__':
c = C()
print c.x
c.x = 10
print c.x
并在控制台中查看以下内容:
pydev debugger: starting
getting
0
File "\test.py", line 55, in <module>
c.x = 10
AttributeError: can't set attribute
我做错了什么?
P.S。:旧式宣言工作正常。
答案 0 :(得分:95)
The documentation says the following关于使用property
的装饰器形式:
请务必为其他函数指定与原始属性相同的名称(在本例中为x。)
我不知道为什么会这样,因为如果你使用property
作为函数来返回属性,可以随意调用方法。
因此您需要将代码更改为以下内容:
@x.setter
def x(self, value):
'setting'
self._x = value
答案 1 :(得分:15)
setter方法必须与getter具有相同的名称。别担心,装饰师知道如何区分它们。
@x.setter
def x(self, value):
...
答案 2 :(得分:6)
当您调用@ x.setter,@ x.getter或@ x.deleter时,您将创建一个新的属性对象并为其指定要装饰的函数的名称。所以真的,最重要的是上次在类定义中使用@ x。* er装饰器时,它具有您实际想要使用的名称。但是,由于这会使您的类命名空间被您希望使用的属性的不完整版本污染,因此最好使用相同的名称来清理它们。
答案 3 :(得分:1)
如果您不想要额外的_x
名称插槽,可以执行以下操作,这是一个复杂的小技巧:
(用Py34测试)
>>> class C(object):
__slots__ = ['x'] # create a member_descriptor
def __init__( self ):
self.x = 0
# or use this if you don't want to call x_setter:
#set_x( self, 0 )
>>> get_x = C.x.__get__ # member_descriptor getter
>>> set_x = C.x.__set__ # member_descriptor setter
>>> # define custom wrappers:
>>> def x_getter( self ):
print('getting')
return get_x( self )
>>> def x_setter( self, value ):
print('setting')
set_x( self, value )
>>> C.x = property( x_getter, x_setter ) # replace the member_descriptor
>>> c = C()
setting
>>> print(c.x)
getting
0
>>> c.x = 10
setting
>>>