Logo

dev-resources.site

for different kinds of informations.

Inline Function โ€” When to use?

Published at
4/29/2024
Categories
kotlin
android
androiddev
Author
Khush Panchal
Categories
3 categories in total
kotlin
open
android
open
androiddev
open
Inline Function โ€” When to use?

In this article, we will investigate the Kotlin inline function in detail and come to a conclusion on when and why to use the inline function.

What are inline functions?

To make the function inline, we will add the inline keyword before fun to convert normal function to inline function

inline fun calculateTime() {
    println("Calculate")
}

fun main() {
    println("Start")
    calculateTime()
    println("End")
}

Decompiled:

public void main() {
   System.out.println("Start");
   System.out.println("Calculate");
   System.out.println("End");
}

As we can see complete body of inline function gets inserted at function call site when decompiled.

Inline function are functions whose body gets copied to the calling place

Advantage of Inline function: No function call overhead โ€” Faster program execution.

Why not make every function inline?

Making Every function inline eventually grow the code as same code will repeat everywhere.

Some will say, if function body is small use inline otherwise not. This is correct upto some extent.

But when we inline a normal function with normal parameters, compiler will give us the below warning

Expected performance impact from inlining is insignificant. Inlining works best for functions with parameters of functional types

So this means expected impact of inlining on performance is negligible as most probably compiler will do it if required.

When do we inline the function?

We generally prefer to inline a function when function take functional type parameters (lambdas)

Letโ€™s dive into the reason for this and how inlining actually help us in this case

In normal case:

fun calculateTime(block: ()->Unit): Long {
    val initialTime = System.currentTimeMillis()
    block.invoke()
    return System.currentTimeMillis() - initialTime
}

fun main() {
    val time = calculateTime {
        println("Hello")
    }
    println(time)
}

Decompiled:

public long calculateTime(Function0 block) {
  long initialTime = System.currentTimeMillis();
  block.invoke();
  return System.currentTimeMillis() - initialTime;
}

public void main() {
  long time = calculateTime(
     new Function() {
        @Override
        public void invoke() {
          System.out.print("Hello");
        }
      }
    );
  System.out.println(time);
}
public interface Function0<out R> : Function<R> {
    public operator fun invoke(): R
}

In Kotlin, function types (lambdas) are converted to object of anonymous/regular classes(Function0) that extend the interface Function

Problem? โ€” If we call this function (calculateTime) 100 times, 100 object of Function class will be created and garbage collected. This affects performance.

Solution? โ€” Use inline for preventing object creation

inline fun calculateTime(block: ()->Unit): Long {
    val initialTime = System.currentTimeMillis()
    block.invoke()
    return System.currentTimeMillis() - initialTime
}

fun main() {
    val time = calculateTime {
        println("Hello")
    }
    println(time)
}

Decompiled:

public void main() {
  long initialTime = System.currentTimeMillis();
  System.out.println("Hello");
  long time = System.currentTimeMillis() - initialTime;
  System.out.println(time);
}

So, the functions that take other functions as arguments are faster when they are inlined (since no Function objects are created).

We should use inline function when we have functional type parameter with small function body.

noinline

What if we have multiple functional type parameter in the inline function and we do not want to inline all the parameters, we can use noinline keyword.

fun main() {
    val time = calculateTime({
        println("Hellow")
    }, {
        println("World")
    })
    println(time)
}

inline fun calculateTime(block1: () -> Unit, noinline block2: () -> Unit): Long {
    val initialTime = System.currentTimeMillis()
    block1.invoke()
    block2.invoke()
    return System.currentTimeMillis() - initialTime
}

Decompiled code:

public static final void main() {
  long initialTime = System.currentTimeMillis();
  System.out.println("Hello");
  Function block = new Function() {
        @Override
        public void invoke() {
          System.out.print("World");
        }
      }
    );
  block.invoke();
  long time = System.currentTimeMillis() - initialTime;
  System.out.println(time);
}

crossinline

crossinline keyword is used to avoid non-local returns.

Letโ€™s understand with example

inline fun calculateTime(block: () -> Unit): Long {
    val initialTime = System.currentTimeMillis()
    block.invoke()
    return System.currentTimeMillis() - initialTime
}

fun main() {
    val time = calculateTime {
        println("Hello")
        return
    }
    println(time)
}

Decompiled:

public static final void main() {
   long initialTime = System.currentTimeMillis();
   System.out.println("Hello");
}

Here we can see that after return, no other statement is written (calculating and printing final time). This is non-local return. To avoid this issue, we can mark the lambda as crossinline.

inline fun calculateTime(crossinline block: () -> Unit): Long {
    val initialTime = System.currentTimeMillis()
    block.invoke()
    return System.currentTimeMillis() - initialTime
}

fun main() {
    val time = calculateTime {
        println("Hello")
        return // This will give compile time error
    }
    println(time)
}

When mark lambda parameter as crossinline, if we add return statement, compiler will give the error (โ€˜returnโ€™ is not allowed here).

Use inline for reified type parameters

What if we want to work with class type directly in Kotlin generics?

fun <T> doSomething(value: T) {
   println("Value: $value") // OK
   println("Type: ${T::class.simpleName}") // Error
}

fun main() {
   doSomething("something")
}

In the above example we get the error

Cannot use 'T' as reified type parameter. Use a class instead

We cannot work with the type directly, because the type argument gets erased at runtime when we pass to the function. So, we cannot possibly know exactly which type we are handling.

Solution? โ€” Use an inline function along with the reified type parameter.

inline fun <reified T> doSomething(value: T) {
   println("Value: $value") // OK
   println("Type: ${T::class.simpleName}") // OK
}

fun main() {
   doSomething("something")
}

In the above example actual type argument will be copied in place of T. So, T::class.simpleName becomes String::class.simpleName.

The reified keyword can only be used with inline functions.

Source Code: Github

Contact Me:
LinkedIn, Twitter

Happy Coding โœŒ๏ธ

Featured ones: