DEV Community

Cover image for Understanding Callable in Python
Yankee Maharjan
Yankee Maharjan

Posted on • Originally published at Medium

Understanding Callable in Python

Functions and classes are the most common things we use in our daily development. We invoke them, pass them around and yet never wonder what makes them so amazing. Well the short answer is callable protocol; for the long answer keep reading!

Definition šŸ“¢

Objects defining the __call__ method is known as callable. Or basically callable is anything you can call using the parenthesis () and pass arguments to it. Yes I am basically talking about a function.

__call__ method šŸ¤™

__call__ is one of the most interesting dunder method in Python. It is what most of the built-in functions make use of. If we peek into the type of some of these built-in functions then we often see the result as a class.

>>> range
<class 'range'>

>>> zip 
<class 'zip'>

>>> int
<class 'int'>
Enter fullscreen mode Exit fullscreen mode

You get the gist. But how is a class acting as a function? I mean you can just zip(iterable1, iterable2) and you get your result without further invoking any other methods of the class.

Just imagine using zip like if it were a normal class.

processor = ['Intel', 'Ryzen', 'Apple Silicon']
year = [2018, 2019, 2020]

zipped = zip(processor, year)
zipped.start()
Enter fullscreen mode Exit fullscreen mode

This definitely isnā€™t intuitive to work with. So how does it work?

Using __call__ method šŸ”Ø

Well behind the scenes itā€™s because of the __call__ method. This method is used in classes if we want the instance of the class to be callable.

What do I mean by that? Letā€™s take an example of a class that prints the square of a number.

class Square:

    def __call__(self, num):
            print(num * num)

>>> sq = Square() # create an instance 
>>> sq(5) # invoke
25
Enter fullscreen mode Exit fullscreen mode

This is similar to executing:

>>> sq = Square()
>>> sq.__call__(5)
25
Enter fullscreen mode Exit fullscreen mode

But with the __call__ dunder method we donā€™t have to do that. We can directly invoke the instance like a normal function.

You may have noticed, we donā€™t need to use the __init__ constructor to pass the value. Since the class instance acts like a function, then it can take arguments like a function without a need of a constructor.

Testing callable šŸ‘Øā€šŸ”¬

A class or a function is callable by default; but the instance is callable with __call__ dunder method only. We can check this by using the callable() function.

Take for example a class without a __call__ method.

    class Random:
        pass

    >>> callable(Random)
    True

    >>> r = Random()
    >>> callable(r)
    False
Enter fullscreen mode Exit fullscreen mode

On the contrary, letā€™s take our Square class from earlier example.

    >>> callable(Square)
    True

    >>> callable(sq)
    True
Enter fullscreen mode Exit fullscreen mode

Conclusion šŸš€

Next time when youā€™re creating a class for something and it needs to return a value in an instant; you can make use of the __call__ method.

I hope this blog has been a help to understand an underlying concept involved in functions and classes. If you have any queries or suggestions, letā€™s discuss them in the comment section.

Top comments (1)

Collapse
 
codeperfectplus profile image
Deepak Raj

Bubble sort implementation using Call function.

class BubbleSort:
    """ BubbleSort Algorithm Implementation in Python 3.0+

        arr : Unorded list
        output : Return list in ascending order.
        time complexity : O(n2)

        Example : 
        >>> sort = BubbleSort()
        >>> sort([4,2,6,5,9,8])
        [2, 4, 5, 6, 8, 9]"""

    def __init__(self):
        print("Bubble Sort Algorithm is Initialized")

    def __call__(self, arr):
        n = len(arr)
        for i in range(n):
            already_sorted = True
            for j in range(n - i - 1):
                if arr[j] > arr[j + 1]:
                    arr[j], arr[j + 1] = arr[j + 1], arr[j]
                    already_sorted = False
            if already_sorted:
                break
        return arr

sort = BubbleSort()
print(sort([10,9, 5, 11, 2]))