Python 属性如何工作?
- 2025-03-14 08:57:00
- admin 原创
- 48
问题描述:
我已经成功使用了 Python 属性,但我不知道它们如何工作。如果我取消引用类外的属性,我只会得到一个类型为 的对象property
:
@property
def hello(): return "Hello, world!"
hello # <property object at 0x9870a8>
但是如果我把属性放在一个类中,行为就会非常不同:
class Foo(object):
@property
def hello(self): return "Hello, world!"
Foo().hello # 'Hello, world!'
我注意到未绑定的Foo.hello
仍然是property
对象,因此类实例化必须发挥魔力,但那是什么魔力呢?
解决方案 1:
正如其他人所指出的,他们使用一种称为描述符的语言特征。
通过类访问时返回实际属性对象的原因Foo.hello
在于属性如何实现__get__(self, instance, owner)
特殊方法:
如果在实例上访问描述符,则该实例将作为适当的参数传递,并且
owner
是该实例的类。当通过类访问时,
instance
则为 None 并且仅owner
传递 。property
对象识别这一点并返回self
。
除了描述符的操作方法之外,另请参阅语言指南中的实现描述符和调用描述符的文档。
解决方案 2:
为了使 @properties 正常工作,类需要是对象的子类。当类不是对象的子类时,第一次尝试访问 setter 时,它实际上会使用较短的名称创建一个新属性,而不是通过 setter 进行访问。
以下无法正常工作。
class C(): # <-- Notice that object is missing
def __init__(self):
self._x = None
@property
def x(self):
print 'getting value of x'
return self._x
@x.setter
def x(self, x):
print 'setting value of x'
self._x = x
>>> c = C()
>>> c.x = 1
>>> print c.x, c._x
1 0
以下将正常工作
class C(object):
def __init__(self):
self._x = None
@property
def x(self):
print 'getting value of x'
return self._x
@x.setter
def x(self, x):
print 'setting value of x'
self._x = x
>>> c = C()
>>> c.x = 1
setting value of x
>>> print c.x, c._x
getting value of x
1 1
解决方案 3:
属性是描述符,而描述符在作为类实例的成员时会表现出特殊行为。简而言之,如果a
是类型的实例A
,并且A.foo
是描述符,则a.foo
相当于A.foo.__get__(a)
。
解决方案 4:
该property
对象仅实现描述符协议:http://docs.python.org/howto/descriptor.html
相关推荐
热门文章
项目管理软件有哪些?
热门标签
曾咪二维码
扫码咨询,免费领取项目管理大礼包!
云禅道AD