为啥 setattr 在绑定方法上失败
Posted
技术标签:
【中文标题】为啥 setattr 在绑定方法上失败【英文标题】:Why does setattr fail on a bound method为什么 setattr 在绑定方法上失败 【发布时间】:2018-09-10 09:24:15 【问题描述】:在下文中,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
【问题讨论】:
为什么需要这个?为什么不使用实例属性?如果您希望每个方法有很多属性,例如self.m_i = 0
或 self.m_data = 'i': 0
。
@eye 感谢您的关注。我正在尝试按照***.com/questions/279561/… 实现静态函数变量,但它也可以与方法一起正常工作。我有大部分的实现,但这是一个症结所在。不过,可能有更好的方法来解决原始问题。
【参考方案1】:
简短的回答:无法将自定义属性添加到绑定方法。
下面是长答案。
在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'
【讨论】:
我得到以下(略有不同)的行为: >>>>print c.m问:“有没有办法在方法上设置一个属性,使它只存在于一个实例上,而不是类的每个实例上?”
答:是的:
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 中以清晰易懂的方式实现它的方式。您使用一个类并使其可调用。在函数上设置属性是可能的,在某些情况下这是一个好主意,但总的来说,它最终只会让人们感到困惑。
【讨论】:
以上是关于为啥 setattr 在绑定方法上失败的主要内容,如果未能解决你的问题,请参考以下文章
Python 元类:为啥在类定义期间不调用 __setattr__ 来设置属性?
为啥 Castle Windsor 拦截器会破坏 C# 动态对象上方法的运行时绑定?