我正在编写一个需要访问私有变量的装饰器,并发现了这种差异。谁能解释一下呢?
(Python 2.5)
命名修改按照预期在类中定义的属性工作:
>>> class Tester(object):
... __foo = "hi"
>>> t = Tester()
>>> t._Tester__foo
'hi'
实例属性不起作用(这是我们应该这样做的方式吗?)
>>> class Tester(object):
... def __init__(self):
... self.__foo = "hi"
>>> t = Tester()
>>> t._Tester__foo
AttributeError: 'Tester' object has no attribute '_Tester__foo'
P.S。 “class attribute”对于这些是正确的词吗?它们不是静态的,但是如果你将其中一个列表或其他可变类型,它就是共享的......
更新
事实上,第二个例子也可以正常工作。这是一个硬件问题(重启帮助)。
答案 0 :(得分:10)
实际上不正确。
名称修改发生在课堂创建时;任何引用受损名称的函数也会被调整。
我无法重现您的示例,至少在Mac上的Python版本2.4,2.5,2.6,3.1和3.2中没有:
>>> class Tester(object):
... def __init__(self):
... self.__foo = "hi"
...
>>> Tester()._Tester__foo
'hi'
>>> Tester().__foo
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
AttributeError: 'Tester' object has no attribute '__foo'
如果您反汇编函数字节码,您可以看到名称也被破坏了:
>>> import dis
>>> dis.dis(Tester.__init__)
3 0 LOAD_CONST 1 ('hi')
3 LOAD_FAST 0 (self)
6 STORE_ATTR 1 (_Tester__foo)
9 LOAD_CONST 0 (None)
12 RETURN_VALUE
我已经检查了compiler source和所有名称是通过mangler运行的,这个代码路径自2002年以来至少保持不变。
是的,类属性和实例属性是正确的术语。类属性始终是共享的,但是将分配给实例上的属性会分配给实例。变异列表或其他可变对象与属性赋值不同。