'无法在Python中使用新样式属性设置属性'

时间:2010-11-15 10:27:04

标签: python

我正在尝试使用新式属性声明:

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。:旧式宣言工作正常。

4 个答案:

答案 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
>>>