In the following, setattr succeeds in the first invocation, but fails in the second, with:
AttributeError: 'method' object has no attribute 'i'
Why is this, and is there a way of setting an attribute on a method such that it will only exist on one instance, not for each instance of the class?
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
The short answer: There is no way of adding custom attributes to bound methods.
The long answer follows.
In Python, there are function objects and method objects. When you define a class, the
defstatement creates a function object that lives within the class' namespace:Function objects have a special
__dict__attribute that can hold user-defined attributes:Method objects are different beasts. They are tiny objects just holding a reference to the corresponding function object (
__func__) and one to its host object (__self__):Method objects provide a special
__getattr__that forwards attribute access to the function object:This is also true for the
__dict__property:Setting attributes follows the default rules, though, and since they don't have their own
__dict__, there is no way to set arbitrary attributes.This is similar to user-defined classes defining
__slots__and no__dict__slot, when trying to set a non-existing slot raises anAttributeError(see the docs on__slots__for more information):