Ever wondered why certain Kotlin functions like repeat(), map(), and filter() can accept suspending functions in their lambda expressions, even when their own signatures aren’t coroutine-aware? 🤔 The answer lies in a powerful Kotlin feature: the inline modifier.
If you’re interested in more Kotlin tips, check out my article on Kotlin 2.0 Updates for Android Developers.
What Are Suspending Functions? ✨
A suspending function is a function marked with the suspend keyword. It can pause its execution without blocking the thread and resume later. These functions are designed to work with coroutines, making asynchronous programming simpler and cleaner.
suspend fun fetchData(): String {
    delay(1000) // simulate asynchronous work
    return "Data fetched"
}
In this example, fetchData() is a suspending function that can only be called from another suspending function or a coroutine.
Inline Functions: A Secret Ingredient in Kotlin 🔍
The inline modifier is crucial to understanding this topic. It essentially tells the compiler to copy the function’s code directly to the call site, reducing function call overhead.

When you call performOperation, the compiler replaces the call with the actual function body.
Solving the Mystery: Why Can `map()`, `filter()`, and `repeat()` Accept Suspending Lambdas? 🔓
So how can functions like map(), filter(), and repeat() accept suspending functions, despite not being coroutine-aware?
The answer is simple: They are marked as inline functions.
The inline keyword allows these functions to accept suspending lambdas as parameters, even when they don’t have the suspend modifier in their own signature. When the function is inlined, the suspending lambda’s code is “embedded” into the call site, making it possible to suspend without issues.
Example: Using Inline with Suspending Functions
suspend fun processItems() {
    val items = listOf(1, 2, 3)
    items.map { item ->
        delay(500) // suspending call inside map
        item * 2
    }
}
Here, map is not marked as suspend, but it still allows a suspending lambda as a parameter. This is because map is an inline function, which allows the suspending behavior to work seamlessly.
Performance Benefits of Inlining
Inlining reduces the function call overhead, leading to performance benefits. However, it can also increase bytecode size since the function code is duplicated at every call site. So, use inline judiciously!
Conclusion: Embrace the Power of Inline and Suspend 🚀
By understanding how inline functions work with suspending lambdas, you can write more efficient and coroutine-friendly code in Kotlin. Inline functions reduce call overhead, and when paired with suspending functions, they open up powerful ways to handle asynchronous programming.
Want to explore more advanced Kotlin topics? Dive into my article on Concurrency in Kotlin Using Coroutines.
Did you like this article? 
You can subscribe to my newsletter below and get updates about my new articles.












Great info 👍. This is a very important point to take into account.
Yes, that’s one of the good things about Kotlin