为什么setattr在绑定方法上失败

时间:2011-10-25 14:56:51

标签: python methods python-3.x setattr

在下文中,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

2 个答案:

答案 0 :(得分:29)

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

接下来的答案很长。

在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(请参阅docs on __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'

答案 1 :(得分:1)

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

答:是的:

class c:

    def m(self):

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

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

链接到的帖子中的函数的静态变量对方法没有用。它在函数上设置一个属性,以便下次调用函数时该属性可用,这样你就可以创建一个计数器或其他东西。

但是方法有一个与它们相关的对象实例(self)。因此,您无需在方法上设置属性,因为您只需在实例上设置它。事实上,这正是实例所针对的。

您链接的帖子显示了如何使用静态变量创建函数。我会说在Python中这样做会被误导。而是看看这个答案:What is the Python equivalent of static variables inside a function?

这是在Python中以清晰且易于理解的方式执行此操作的方法。您使用类并使其可调用。在函数上设置属性是可能的,并且可能存在这样一个好主意的情况,但总的来说它最终会让人感到困惑。