Inline Functions

https://kotlinlang.org/docs/inline-functions.html
Inline Functions

Способ оптимизации производительности за счет устранения накладных расходов, связанных с вызовом функций, особенно при использовании лямбд. Основная идея состоит в том, что тело функции и переданные лямбды будут вставлены («инлайнятся») в место вызова функции, что исключает создание дополнительных объектов и вызовов. В Kotlin лямбды активно используются, но их вызов обычно создает накладные расходы на создание дополнительных объектов (анонимных классов или функций), что может сказаться на производительности в критичных участках кода (например, в циклах или высоконагруженных операциях). inline-функции помогают решить эту проблему.

Когда инлайнить функцию

Использование лямбд как аргументов. Если функция принимает лямбды в качестве аргументов, инлайнинг может значительно повысить производительность. Без инлайнинга для каждой лямбды будет создаваться объект, что требует дополнительной памяти и времени на его создание. inline-функция вставляет тело лямбды напрямую в место вызова.

// В этом случае лямбда action не требует создания отдельного объекта при вызове performOperation().
inline fun performOperation(action: () -> Unit) {
    action()
}

Избегание накладных расходов на вызов функции. Если функция вызывается очень часто или выполняется в цикле, накладные расходы на вызов функции могут стать значительными. Инлайнинг может помочь избавиться от этих затрат, так как вместо вызова функции её код просто подставляется на место вызова.

// В цикле или при частом вызове логирования инлайнинг устранит накладные расходы на вызов функции.
inline fun logMessage(message: String) {
    println(message)
}

Использование reified типов. Как уже упоминалось, ключевое слово reified работает только с inline-функциями. Если функция должна манипулировать типом generics на этапе выполнения, её нужно инлайнить.

// Без инлайнинга нельзя будет сохранить информацию о типе T для проверки на этапе выполнения.
inline fun <reified T> isType(value: Any): Boolean {
    return value is T
}

Код маленького объёма. Если функция имеет небольшое тело, инлайнинг может быть выгоден, так как вставка тела функции не приведёт к существенному увеличению объёма сгенерированного байт-кода, но избавит от накладных расходов на вызов.

inline fun add(a: Int, b: Int): Int = a + b
Когда не стоит инлайнить

Крупные функции: Если функция большая, инлайнинг может привести к значительному увеличению объёма байт-кода, что замедлит компиляцию и увеличит размер программы.

Рекурсивные функции: Рекурсивные функции не могут быть inline-функциями, так как это приведёт к бесконечной подстановке вызовов самой функции в себя.

Редко вызываемые функции: Если функция вызывается редко, инлайнинг может быть излишним.

inline

Заменяет вызовы функции её телом на этапе компиляции, устраняя накладные расходы. Это ускоряет выполнение, особенно с лямбдами, так как они не создают объекты.

inline fun example(block: () -> Unit) {
    block()
}
noinline

Запретить встраивание лямбды в месте вызова.

inline fun inlineExample(inlined: () -> Unit, noinline notInlined: () -> Unit) {
    inlined() // будет инлайнятся
    notInlined() // не будет инлайнятся
}
crossinline

Запретить вызов return в лямбде.

inline fun crossinlineExample(crossinline block: () -> Unit) {
    Thread {
        block() // не позволит использовать return для выхода из всей функции
    }.start()
}
reified

Используется в сочетании с inline-функциями, чтобы позволить работать с типами generics на этапе выполнения (runtime). В обычных generic-функциях информация о типах стирается во время компиляции из-за механизма type erasure в JVM, но с помощью reified можно сохранить тип и оперировать им в коде. Благодаря встраиванию, компилятор может «развернуть» функцию и получить конкретный тип, переданный в качестве аргумента, и использовать его в коде. Это делает возможным использование таких операций, как проверка типа is или создание объектов newInstance, которые требуют информации о типах во время выполнения.

inline fun <reified T> checkType(value: Any) {
    if (value is T) {
        println("This is of type ${T::class}")
    } else {
        println("This is not of type ${T::class}")
    }
}
Inline Properties

Представляют собой механизм, позволяющий инлайнить геттеры и сеттеры свойств, чтобы избежать накладных расходов, связанных с вызовом методов. Это работает аналогично inline-функциям, и применяется в случаях, когда геттер или сеттер выполняет простые операции, такие как возвращение значения или присвоение.

val password: String
    inline get() = "qwerty"

var complexProperty: Int
    inline get() {
        val x = 2
        val y = 4
        return x + y
    }
    inline set(value) {
        val adjustedValue = value + 10
        println("Setting adjusted value $adjustedValue")
    }
Вопросы на собесе (15)
  • reified (4)
    1. Для чего используется reified?

      Используется в обобщённых функциях для сохранения информации о типе во время выполнения, позволяя напрямую работать с типом-параметром, чего обычно нельзя делать из-за стирания типов.

    1. Почему reified можно использовать только с inline-функциями?

      Позволяет сохранить типы дженериков, которые обычно стираются во время выполнения.

    1. Почему reified позволяет понять тип объекта?

      Потому что в inline-функциях код внедряется прямо в место вызова, и благодаря этому типы дженериков не стираются на этапе компиляции.

    1. Какую проблему помогает решить ключевое слово reified?

      Memory leak.

      Стирение дженерик-типов.

      Оптимизация использования памяти.

  • Другие (11)
    1. Что такое-inline функции в Kotlin и зачем они нужны?

      inline-функции в Kotlin позволяют оптимизировать код, заменяя вызовы функций их телами во время компиляции. Это снижает накладные расходы на вызов функции, особенно при использовании лямбд, избегая создания дополнительных объектов и контекстов вызовов.

    1. Какие по размеру функции рекомендуется инлайнить?

      Маленькие по объёму: содержит минимальное количество инструкций, например, простые математические операции, проверки условий, или короткие выражения.

    1. Для чего используется noinline?

      Предотвращает встраивание лямбда-выражений в код, позволяя передавать их как обычные функции.

    1. Для чего используется crossinline?

      Предотвращает неявный возврат из лямбда-выражения в inline-функциях через return, обеспечивая корректное управление потоком.

    1. Минусы использования inline-функций? (Когда не стоит инлайнить)

      Увеличивается размер байт-кода. Возрастает сложность отладки так как код становится менее читаемым.

    1. Как понять, что функцию нужно инлайнить?

      Функцию следует инлайнить, если она маленькая и часто вызывается, так как это снижает накладные расходы на вызов функции и может улучшить производительность. Также инлайнинг полезен для функций, которые принимают лямбды, чтобы избежать создания дополнительных объектов.

    1. За счет чего выполняется оптимизация в inline-функциях?

      За счет замены вызова функции непосредственно вставленным в код телом функции (сокращение количества вызовов). Это позволяет избежать накладных расходов на вызов функции и может улучшить производительность.

    1. Что такое non-local return?

      Это возможность выйти из внешней функции изнутри lambda или inline-функции. В Kotlin это позволяет возвращать значения не только из inline-функции, но и из её enclosing функции, что улучшает управление потоком выполнения кода.

    1. Можно ли вызвать inline-функцию из Java?

      Нет, так как она реализуется на уровне вызова в Kotlin-коде, и при компиляции в байткод её тело вставляется в места вызова.

    1. Что будет если заинлайнить все функции в приложении?

      Если заинлайнить все функции в приложении, размер байткода значительно увеличится, что приведет к большему потреблению памяти и более длительному времени компиляции. Это может также снизить производительность за счёт увеличения нагрузки на кеш процессора и ухудшения управляемости кода, так как инлайн-подстановка оправдана только для небольших, часто вызываемых функций.

    1. Действие какого модификатора описано ниже? Модификатор влияет и на функцию, и на лямбду, переданную ей: они обе будут встроены вместо вызова

      noinline

      crossinline

      reified

      inline