Search code examples
pythoninstance

Why won't dynamically adding a `__call__` method to an instance work?


In both Python 2 and Python 3 the code:

class Foo(object):
    pass

f = Foo()
f.__call__ = lambda *args : args

f(1, 2, 3)

returns as error Foo object is not callable. Why does that happen?

PS: With old-style classes it works as expected.

PPS: This behavior is intended (see accepted answer). As a work-around it's possible to define a __call__ at class level that just forwards to another member and set this "normal" member to a per-instance __call__ implementation.

Python 3.10.8 (main, Nov  1 2022, 14:18:21) [GCC 12.2.0]
Type 'copyright', 'credits' or 'license' for more information
IPython 8.7.0 -- An enhanced Interactive Python. Type '?' for help.

In [1]: class Foo:
   ...:     def __call__(self, *args, **kwargs):
   ...:         return self.call(*args, **kwargs)
   ...: 

In [2]: f = Foo()

In [3]: f.call = lambda *args, **kwargs: (args, kwargs)

In [4]: f(1,2,3)
Out[4]: ((1, 2, 3), {})

Solution

  • Double-underscore methods are always looked up on the class, never the instance. See Special method lookup for new-style classes:

    For new-style classes, implicit invocations of special methods are only guaranteed to work correctly if defined on an object’s type, not in the object’s instance dictionary.

    That's because the type might need to support the same operation (in which case the special method is looked up on the metatype).

    For example, classes are callable (that's how you produce an instance), but if Python looked up the __call__ method on the actual object, then you could never do so on classes that implement __call__ for their instances. ClassObject() would become ClassObject.__call__() which would fail because the self parameter is not passed in to the unbound method. So instead type(ClassObject).__call__(ClassObject) is used, and calling instance() translates to type(instance).__call__(instance).

    To work around this, you could add a __call__ method to the class which checks for a __call__ attribute on the class, and if there, calls it.