【发布时间】:2012-05-20 23:02:35
【问题描述】:
这是一个关于实例变量的 Python 新手问题。
考虑以下 Python 2.7 类定义:
class Foo(object):
a = 1
def __init__(self):
self.b = 2
def __repr__(self):
return "%s" % self.__dict__
现在,当我创建 Foo 的实例时,Foo.__dict__ 包含 b,但不包含 a。
>>> x=Foo()
>>> x
{'b': 2}
>>> dir(x)
['__class__', '__delattr__', '__dict__', '__doc__', '__format__',
'__getattribute__', '__hash__', '__init__', '__module__', '__new__',
'__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__',
'__str__', '__subclasshook__', '__weakref__', 'a', 'b']
>>> x.__dict__
{'b': 2}
在这里,我认为我对 Python 之道有很好的了解。
x.a 和 x.b 有什么区别?据我所知,它们都是实例变量。
编辑:好的,重新阅读Python docs 我看到Foo.a 是一个类属性,而不是一个实例变量。嗯...我想混乱来自这样一个事实,即我可以为 x.a 分配一个新值,而新值只影响 x 实例——我想我现在在Foo.a 属性:
>>> y=Foo()
>>> y.a = 2
>>> y
{'a': 2, 'b': 2}
>>> x
{'b': 2}
>>> x.a
1
>>> z=Foo()
>>> z
{'b': 2}
>>> z.a
1
>>> Foo.a
1
>>> x.a
1
>>> y.a
2
所以,现在我覆盖了Foo.a 的先前值,它会影响所有没有别名为Foo.a 的Foo 实例:
>>> Foo.a=999
>>> x.a
999
>>> y.a
2
【问题讨论】:
-
还要注意,过多地依赖
__dict__并不是一个好习惯:使用__slots__的类和扩展类(来自C 或Cython)也不会把所有东西都放在那里。大多数用途可以用getattr、setattr、hasattr和dir的某种组合代替。 -
@Dougal:感谢您的建议——我在 StackOverflow 上的另一个答案中看到了这一点,但这是很好的强化。
标签: python class python-2.7