【问题标题】:Python class instance __dict__ doesn't contain all instance variables. Why?Python 类实例 __dict__ 不包含所有实例变量。为什么?
【发布时间】: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.ax.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.aFoo 实例:

>>> Foo.a=999
>>> x.a
999
>>> y.a
2

【问题讨论】:

  • 还要注意,过多地依赖__dict__ 并不是一个好习惯:使用__slots__ 的类和扩展类(来自C 或Cython)也不会把所有东西都放在那里。大多数用途可以用getattrsetattrhasattrdir 的某种组合代替。
  • @Dougal:感谢您的建议——我在 StackOverflow 上的另一个答案中看到了这一点,但这是很好的强化。

标签: python class python-2.7


【解决方案1】:

您的 a 不是实例变量。您将其定义为类的一部分。

>>> class Foo(object):
...    a = 1
...
>>> Foo.a
1

如果你想要一个实例变量,你应该把它放在__init__ 方法中,因为这个方法在你的对象被创建时被调用。

【讨论】:

    【解决方案2】:

    a 不是实例属性,而是类属性。

    【讨论】:

    • 是的,但是如果我为x.a 分配一个新值,它只会影响x 实例。如果是类属性,为什么更改x.a 的值不会影响Foo 的所有实例?
    • 因为你在实例上分配了它。
    • 是的,我现在明白了。我在最初阅读文档时错过了这一点。感谢您的帮助!
    【解决方案3】:

    这对您有进一步帮助吗?

    >>> class X(object):
    
        def __getattribute__(self, name):
            print name
            return object.__getattribute__(self, name)
    
    
    >>> l = dir(X())
    __dict__
    __members__
    __methods__
    __class__
    >>> l
    ['__class__', '__delattr__', '__dict__', '__doc__', '__format__', '__getattribute__', '__hash__', '__init__', '__module__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__']
    

    【讨论】:

      猜你喜欢
      • 2020-04-05
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2013-11-02
      • 2021-11-10
      • 2016-05-27
      • 1970-01-01
      相关资源
      最近更新 更多