18

在下文setattr中,第一次调用成功,但第二次调用失败,其中:

AttributeError: 'method' object has no attribute 'i'

为什么会这样,有没有办法在方法上设置一个属性,使其只存在于一个实例上,而不是类的每个实例上?

class c:

    def m(self):

        print(type(c.m))
        setattr(c.m, 'i', 0)

        print(type(self.m))
        setattr(self.m, 'i', 0)

Python 3.2.2

4

2 回答 2

33

简短的回答:无法将自定义属性添加到绑定方法。

长答案如下。

在 Python 中,有函数对象方法对象。当您定义一个类时,该def语句会创建一个位于该类的命名空间内的函数对象:

>>> class c:
...     def m(self):
...         pass
...
>>> c.m
<function m at 0x025FAE88>

函数对象有一个特殊的__dict__属性,可以保存用户定义的属性:

>>> c.m.i = 0
>>> c.m.__dict__
{'i': 0}

方法对象是不同的野兽。它们是微小的对象,只是持有对相应函数对象 ( __func__) 的引用和一个对其宿主对象 ( __self__) 的引用:

>>> c().m
<bound method c.m of <__main__.c object at 0x025206D0>>
>>> c().m.__self__
<__main__.c object at 0x02625070>
>>> c().m.__func__
<function m at 0x025FAE88>
>>> c().m.__func__ is c.m
True

方法对象提供了__getattr__将属性访问转发到函数对象的特殊功能:

>>> c().m.i
0

这也适用于__dict__财产:

>>> c().m.__dict__['a'] = 42
>>> c.m.a
42
>>> c().m.__dict__ is c.m.__dict__
True

但是,设置属性遵循默认规则,并且由于它们没有自己的__dict__,因此无法设置任意属性。

这类似于用户定义的类定义__slots__并且没有__dict__插槽,当尝试设置一个不存在的插槽时会引发一个(有关更多信息,AttributeError请参阅文档):__slots__

>>> class c:
...     __slots__ = ('a', 'b')
...
>>> x = c()
>>> x.a = 1
>>> x.b = 2
>>> x.c = 3
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: 'c' object has no attribute 'c'
于 2011-10-25T15:26:53.883 回答
2

问:“有没有办法在方法上设置一个属性,使它只存在于一个实例上,而不是类的每个实例上?”

答:是的:

class c:

    def m(self):

        print(type(c.m))
        setattr(c.m, 'i', 0)

        print(type(self))
        setattr(self, 'i', 0)

您链接到的帖子中函数的静态变量对方法没有用。它在函数上设置一个属性,以便下次调用该函数时该属性可用,因此您可以制作一个计数器或诸如此类的东西。

但是方法有一个与之关联的对象实例(self)。因此,您无需在方法上设置属性,因为您只需在实例上设置它即可。这实际上正是该实例的用途。

您链接到的帖子显示了如何使用静态变量创建函数。我会说在 Python 中这样做会被误导。相反,看看这个答案:什么是函数内部静态变量的 Python 等价物?

这就是在 Python 中以清晰易懂的方式实现它的方式。您使用一个类并使其可调用。在函数上设置属性是可能的,并且在某些情况下它可能是个好主意,但总的来说它只会让人们感到困惑。

于 2011-10-26T01:17:51.273 回答