I bumped into this behaviour when trying to get class-decorators and method-decorators to play nicely together. Essentially, the method decorators would flag some of the methods as special with some dummy value, and the class decorator would come by after and fill in the value later. This is a simplified example
>>> class cow: >>> def moo(self): >>> print 'mooo' >>> moo.thing = 10 >>> >>> cow.moo.thing 10 >>> cow().moo.thing 10 >>> cow.moo.thing = 5 AttributeError: 'instancemethod' object has no attribute 'thing' >>> cow().moo.thing = 5 AttributeError: 'instancemethod' object has no attribute 'thing' >>> cow.moo.__func__.thing = 5 >>> cow.moo.thing 5
Does anyone know why cow.moo.thing = 5
does not work, even though cow.moo.thing
quite clearly gives me 10? And why cow.moo.__func__.thing = 5
works? I have no idea why it does, but in randomly fiddling with stuff in the dir(cow.moo)
list trying to get something to work it suddenly did, and i have no idea why.
Everything in Python is an object, and almost everything has attributes and methods.
An instance attribute is a Python variable belonging to one, and only one, object. This variable is only accessible in the scope of this object and it is defined inside the constructor function, __init__(self,..) of the class.
Adding attributes to a Python class is very straight forward, you just use the '. ' operator after an instance of the class with whatever arbitrary name you want the attribute to be called, followed by its value.
For attribute lookup, Python is automatically using the real function attached to the instance method for you.
For attribute setting, it is not.
They are two separate operations depending on which side of the statement you're on, even though they both use the .
operator.
When you access an instance method's __func__
, you're manually accessing the real function that actually has the moo
attribute.
In Python 3 this will work as you would like / expect as methods are basically just functions.
If you are looking to modify the function attributes of both functions and instance methods from C then you have to check the type of callable you have.
So assuming you have a PyObject of some type of callable you can check it like this:
PyObject *callable; // set to something callable PyObject *setting; // set to something if(PyMethod_Check(callable)){ PyObject_SetAttrString(PyMethod_Function(callable),"attribute",setting); }else{ PyObject_SetAttrString(callable,"attribute",setting); } ... // and the inverse if(PyMethod_Check(callable){ if(PyObject_HasAttrString(PyMethod_Function(callable),"attribute")){ PyObject_DelAttrString(PyMethod_Function(callable),"attribute"); } }else{ if(PyObject_HasAttrString(callable,"attribute")){ PyObject_DelAttrString(callable,"attribute"); } }
Now the code agf pointed out works from within Python for instance methods. If I just try to set the attribute of the instance method it would not find the attribute no matter how I tried to access it from Python.
I ran into this issue and Li Haoyi's question with agf's answer helped me understand what needed to be changed. I figured someone will find this question and answer again while looking for how to solve this issue through C.
Edit: Note: This is for Python 2.7.x. Python 3.x uses different function calls.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With