KotlinMiddleCoding
Что такое scope functions в Kotlin (let, run, apply, also, with) и когда использовать каждую?
let/run/apply/also/with — функции-области видимости: различаются тем, как передаётся объект (this или it) и что возвращают (объект или результат лямбды). Выбор зависит от цели: трансформация, конфигурация или side-effect.
Обзор scope functions
Все пять функций принимают лямбду и выполняют её в контексте объекта. Различия — в двух осях: как объект виден внутри лямбды (this или it) и что возвращается (результат лямбды или сам объект).
| Функция | Ссылка | Возвращает | Extension |
|---|---|---|---|
| let | it | результат лямбды | да |
| run | this | результат лямбды | да |
| apply | this | объект | да |
| also | it | объект | да |
| with | this | результат лямбды | нет (аргумент) |
let — трансформация и null-safe цепочки
val name: String? = "Alice"
val length = name?.let { it.length } // Int? = 5
// Трансформация значения
val upper = "hello".let { it.uppercase() } // HELLO
run — вычисление с контекстом this
val result = StringBuilder().run {
append("Hello")
append(", World")
toString() // возвращается из run
}
println(result) // Hello, World
apply — конфигурация объекта (Builder-pattern)
data class Config(var host: String = "", var port: Int = 0)
val cfg = Config().apply {
host = "localhost" // this — Config
port = 8080
}
// cfg.host == "localhost", cfg.port == 8080
also — side-effect без изменения объекта
val list = mutableListOf(1, 2, 3)
.also { println("Before: $it") } // логирование
.also { it.add(4) } // мутация через it
println(list) // [1, 2, 3, 4]
with — операции над объектом без extension
val stats = with(listOf(1, 2, 3, 4)) {
"min=${min()}, max=${max()}, sum=${sum()}"
}
println(stats) // min=1, max=4, sum=10
Практические правила выбора
let— null-check + трансформация:obj?.let { transform(it) }apply— конфигурация нового объекта (возврат самого объекта удобен в builder-цепочках)also— логирование, assertions или другие side-effect без изменения цепочкиrun— замена нескольких строк в блоке с общим контекстомthiswith— когда объект уже есть в переменной и нужно несколько операций без повторного указания имени
Подводные камни
- Вложенные scope functions с одним и тем же именем
itтеряют читаемость — всегда переименовывайте:let { user -> user.name.let { name -> … } } applyвозвращает объект, а не результат лямбды — если забыть, можно получить неожиданный тип.withне является extension: нельзя использовать в цепочке через точку.- Чрезмерное использование создаёт «лесенку» из отступов, которая сложнее читается, чем прямой код.
- В
applyссылки наthisмогут случайно захватить внешнийthisкласса, а не объект — нужна осторожность внутри методов класса. - Возврат
Unitизletтехнически возможен, но это антипаттерн — используйтеalso. runбез объекта-получателя — просто блок кода: легко перепутать две перегрузки.
Common mistakes
- Объяснять «scope functions» только как синтаксис и не описывать поведение runtime/compiler.
- Игнорировать важный риск: Цепочки scope functions быстро становятся нечитаемыми, если скрывают имя receiver-а и смешивают mutation с вычислением.
- Давать пример без edge case: отмены, null, recomposition, platform boundary или ошибки.
What the interviewer is testing
- Формулирует суть темы «scope functions» своими словами и связывает ее с кодом.
- Называет механизм: Выбор функции должен отражать намерение: apply для конфигурации объекта, also для side effect, let для nullable chaining или преобразования.
- Видит production-последствие: Цепочки scope functions быстро становятся нечитаемыми, если скрывают имя receiver-а и смешивают mutation с вычислением.