[Pyrex] pyrex functions to replace a method (Re: replace a	method in class: how?)
    Lenard Lindstrom 
    len-l at telus.net
       
    Wed Jun 28 18:40:15 UTC 2006
    
    
  
On 28 Jun 2006 at 7:02, Brian Blais wrote:
> Greg Ewing wrote:
> > Brian Blais wrote:
> >> I have found a very similar problem trying to replace a method using a 
> >> function defined in pyrex.
> > 
> > 
> > What *should* work is to define the method inside a
> > class in Pyrex (plain class, not extension type) and
> > extract it out of the class's __dict__. That's because
> > Pyrex pre-wraps a function defined in a class in an
> > unbound method object before putting it in the class.
> > 
> 
> So I tried:
> 
> #---------------------------------------------------------------------------------
> 
> #module_pyrex.pyx
> 
> class update_funcs:
> 
>      def pyrex_update_within_class(self,val):
>          print "pyrex module within class",val
> 
> 
> #---------------------------------------------------------------------------------
> 
> #(adding to test_replace_method.py)
> 
> This.update4=module_pyrex.update_funcs.__dict__['pyrex_update_within_class']
> 
> t.update4('pyrex within class') # doesn't work
> 
> #---------------------------------------------------------------------------------
> 
> and get:
> 
> TypeError: unbound method pyrex_update_within_class() must be called with 
> update_funcs instance as first argument (got str instance instead)
> 
> 
> did I do this wrong?
Nothing. Python's method type is just too specialized to work as Greg 
suggested. You will have to provide your own method descriptor. This 
works:
#--------------------------------------------------------------------
#module_pyrex.pyx
cdef extern from "python.h":
    object PyMethod_New(object func, object self, object cls)
# Yes, this has to be an extension type in Pyrex.
cdef class InstanceMethod:
    cdef object fn
    def __init__(self, fn):
        self.fn = fn
    def __get__(self, instance, owner):
        return PyMethod_New(self.fn, instance, owner)
def pyrex_update_within_class(self,val):
     print "pyrex module within class",val
pyrex_update_within_class = InstanceMethod(pyrex_update_within_class)
#--------------------------------------------------------------------
#(Make this replacement in main py module of original posting)
This.update3=module_pyrex.pyrex_update_within_class
#--------------------------------------------------------------------
InstanceMethod is minimal. Additions may be to make the fn attribute 
readable and a __call__ method that calls fn directly.
Lenard Lindstrom
<len-l at telus.net>
    
    
More information about the Pyrex
mailing list