How to get a reference of the 'owner' class to which a method belongs in Python 3.X?

Cosmia Luna cosmius at gmail.com
Sat Mar 17 06:04:58 EDT 2012


On Saturday, March 17, 2012 5:25:06 PM UTC+8, Peter Otten wrote:
> Cosmia Luna wrote:
> 
> > I'm porting my existing work to Python 3.X, but...
> > 
> > class Foo:
> >     def bar(self):
> >         pass
> > 
> > mthd = Foo.bar
> > 
> > assert mthd.im_class is Foo # this does not work in py3k
> > 
> > So, how can I get a reference to Foo? This is important when writing
> > decorators, the only way I can think out is:
> > 
> > class Foo:
> >     def bar(self):
> >         'Foo' # manually declare the owner class
> >         pass
> > 
> > mthd = Foo.bar
> > 
> > assert mthd.__globals__[mthd.__doc__] is Foo # this works
> > 
> > class Child(Foo):
> >     def bar(self):
> >         'Child' # I have to override all method defined by bar but do
> >         nothing pass
> > 
> > child_mthd = Child.bar
> > 
> > assert child_mthd.__globals__[child_mthd.__doc__] is Child # this works
> > 
> > But the code above is quite ugly and abuses the __doc__. Is there any
> > equivalent in py3k of im_class?
> 
> class set_class:
>     def __init__(self, method):
>         self.method = method
>     def __get__(self, instance, class_):
>         if instance is None:
>             method = self.method
>             def f(*args, **kw):
>                 return method(*args, **kw)
>             f.im_class = class_
>             f.__name__ = method.__name__
>             return f
>         return self.method.__get__(instance, class_)
>     
> class Foo:
>     def __init__(self, name):
>         self.name = name
>     @set_class
>     def bar(self):
>         print("%s says hello from bar()" % self)
>     def __str__(self):
>         return self.name
> 
> class Bar(Foo):
>     pass
> 
> assert Foo.bar.im_class is Foo
> assert Bar.bar.im_class is Bar
> 
> Foo("Fred").bar()
> Foo.bar(Foo("Fred"))
> 
> Bar("Fred").bar()
> Bar.bar(Bar("Fred"))
> 
> The cleaner approach is probably:
> 
> Rule('/<action>/', endpoint=(RootController, RootController.otheraction))
> ...
> Controller, method = endpoint
> controller = Controller(Request(environ))
> ...
> method(controller)


That's exactly what I want, and I think you're right, the approach below is cleaner.
Rule('/<action>/', endpoint=(RootController, RootController.otheraction)).

Thanks a lot.
Cosmia



More information about the Python-list mailing list