how to find out if an object is a class?

S

szczepiq

Pardon me for most likely a dummy question but how do I find out if an
object is a class?

I need something like that:

def foo(self, obj):
if (obj is a class):
some stuff
 
T

Terry Reedy

Wojtek said:
Dnia Thu, 7 Aug 2008 14:36:37 -0700 (PDT), szczepiq napisa�(a):


Use types.ClassType:

... pass
...

That is only true and only works for 2.x old-style classes and not for
2.x new-style classes and all 3.0 classes, for which isinstance(Q,type)
is True.
<type 'type'>
 
C

Carl Banks

That is only true and only works for 2.x old-style classes and not for
2.x new-style classes and all 3.0 classes, for which isinstance(Q,type)
is True.

isinstance(Q,type) is also true for built in types and C extension
types, which may or may not be what the OP wants.

The most accurate way I can think of to check for a class defined in
Python is to test the type's tp_flags field for Py_TPFLAGS_HEAPTYPE
bit, but I don't know of any simple way to check for it from Python.
It's still possible for it to fail since someone could create heap
types in C though I'd expect that's very rare.


Carl Banks
 
T

Terry Reedy

Carl said:
isinstance(Q,type) is also true for built in types and C extension
types

That is rather the point of new and improved classes -- that the
implementation language of a class be pretty much irrelevant from the
user api viewpoint ;-)
 
S

Stefan Behnel

Ben said:
Presumably you want to know whether it's a class in order to use it
for instantiating it. It is usually more Pythonic to use the object as
intended, and allow the object itself to tell you (via exceptions)
when it's not behaving as you expect.

This also allows passing a factory function instead of a class, BTW.

I recently had the reverse case that a (stupidly implemented) extension module
required a callback function and I wanted to pass a function wrapped in a
wrapper object. That failed, because it specifically checked for the argument
being a function, not just a callable object. I had to pull quite a number of
tricks to reimplement the wrapper class as a function (thank god, it's Python!).

Stefan
 
M

Miles

I recently had the reverse case that a (stupidly implemented) extension module
required a callback function and I wanted to pass a function wrapped in a
wrapper object. That failed, because it specifically checked for the argument
being a function, not just a callable object. I had to pull quite a number of
tricks to reimplement the wrapper class as a function (thank god, it's Python!).

You really only needed one trick:

def functionize(callable):
return lambda *args, **kwargs: callable(*args, **kwargs)

:)

-Miles
 
C

Christian Heimes

szczepiq said:
Pardon me for most likely a dummy question but how do I find out if an
object is a class?

For God's sake don't reinvent the wheel! The 'inspect' module (part of
the Python standard library) has a functions isclass(). It does the
proper tests for new style and old style classes.

import inspect
inspect.isclass(something)

Christian
 
S

Stefan Behnel

Miles said:
You really only needed one trick:

def functionize(callable):
return lambda *args, **kwargs: callable(*args, **kwargs)

Congratulations, you found the trivial case.

Stefan
 
S

Steven D'Aprano

Congratulations, you found the trivial case.

What other cases are there? It takes any callable, and returns a function
that calls the callable. What else do you need? This is not a rhetorical
question.

The above works as expected for classes and methods:
x = functionize(float)
type(x)
x('123') 123.0
L = [1,2,3]
y = functionize(L.append)
y(44)
L
[1, 2, 3, 44]

It even works for recursive functions:
.... if n <= 1: return 1
.... else: return n*fact(n-1)
....120


I haven't tested it on classmethods, staticmethods, or instances with a
__call__ method, but I see no reason why it wouldn't work with them. What
am I missing?
 
S

Stefan Behnel

Steven said:
Congratulations, you found the trivial case.

What other cases are there? It takes any callable, and returns a function
that calls the callable. What else do you need? [...] What am I missing?

The words "stupidly implemented" above? :)

I have to set the callback in more than one place and (believe it or not) the
library behaves (slightly) different when you pass different callbacks. The
right way to use the above approach would be to wrap the callback right before
passing it into the library - which I can't do, as that would give me a
different function object each time. Also, most of the time I actually pass a
function, except in one case where I need a wrapper for an existing function.
So the solution I chose was to change the original wrapper class itself
instead of re-wrapping it, so that I get the expected object right away.

As usual, it's all about the details.

Stefan
 

Ask a Question

Want to reply to this thread or ask your own question?

You'll need to choose a username for the site, which only take a couple of moments. After that, you can post your question and our members will help you out.

Ask a Question

Members online

No members online now.

Forum statistics

Threads
474,000
Messages
2,570,252
Members
46,848
Latest member
CristineKo

Latest Threads

Top