Skip to content

2.1 Coroutines

Bora Canbula edited this page Dec 2, 2023 · 4 revisions

Coroutines are a generalization of subroutines (functions) used for cooperative multitasking. Unlike functions that return once and do not maintain state, coroutines can pause their execution and later continue from where they left off.

import time
import asyncio

# traditional function
def my_function():
    print("Start of the function")
    time.sleep(1)
    print("End of the function")

# coroutine
async def my_coroutine():
    print("Start of the coroutine")
    await asyncio.sleep(1)
    print("End of the coroutine")

my_function()  # call function
asyncio.run(my_coroutine())  # call coroutine

Output:

Start of the function
End of the function
Start of the coroutine
End of the coroutine

The performance improvement comes from awaiting more than one coroutine. So you can start more than one coroutine at once to solve multiple IO-bound problems, such as fetching some data, reading some files, etc.

async def coroutine_1():
    print("Start of coroutine_1")
    await asyncio.sleep(1)
    print("End of coroutine_1")

async def coroutine_2():
    print("Start of coroutine_2")
    await asyncio.sleep(1)
    print("End of coroutine_2")

# call coroutines with asyncio.gather
async def main():
    coroutines = [coroutine_1(), coroutine_2()]
    await asyncio.gather(*coroutines)

asyncio.run(main())

Output:

Start of coroutine_1
Start of coroutine_2
End of coroutine_1
End of coroutine_2

Awaitable Class

Like naming functions as callables, the term awaitable is introduced with coroutines. Therefore similar to defining __call__ method of a class to make it callable, we define __await__ method of a class to make it awaitable. You can call these type of classes directly, or you can create an object from them.

class AwaitableClass:
    def __await__(self):
        print("Awaiting")
        yield
        print("Done")

async def main():
    print("Starting main")
    await AwaitableClass()
    awaitable_object = AwaitableClass()
    await awaitable_object
    print("Ending main")

asyncio.run(main())