На прошлой неделе при поддержке Redmadrobot SPB в рамках SPB Kotlin User Group прошла встреча со Станиславом Ерохиным, разработчиком из JetBrains. На встрече он поделился информацией о разрабатываемых возможностях следующей мажорной версии Котлина (под номером 1.3).


В этой статье мы подытожим полученный материал, расскажем о планах.


Важно. Дизайн всех фич, о которых пойдет речь, может быть изменен до неузнаваемости. Все, что ниже — это текущие планы команды, но никаких гарантий разработчики не дают.


Посмотреть доклад можно по ссылке на YouTube.


Введение


В ближайшей мажорной версии Kotlin (1.3) ожидается много изменений. Рассказывали нам только о некоторых из них.


Основные направления:


  • Корутины
  • Inline классы
  • Беззнаковая арифметика
  • Default методы для Java
  • Метаинформация для DFA (Data Flow Analysis) и смарткастов
  • Аннотации для управления type inference
  • SAM для Java методов
  • SAM для Kotlin методов и интерфейсов
  • Smart inference для билдеров
  • Изменения в схеме компилятора

Корутины


В версии 1.3 планируется долгожданный релиз корутин в Котлине, в результате чего они переберутся из пакета kotlin.coroutines.experimental в kotlin.coroutines.


Позже планируется выпустить support library для поддержки существующего API корутин.


Основные изменения в корутинах


  • JetBrains активно работают над производительностью. Например, планируется, что в новой версии не будут генерироваться state-machine там, где они не нужны. За счет этого количество генерируемых объектов уменьшится
  • Ведутся работы над полноценной поддержкой suspend для inline функций
  • Добавится поддержка callable reference для suspend функций
  • Изменится интерфейс Continuation

В данный момент Continuation содержит два метода, которые позволяют пробросить либо результат, либо ошибку.


interface Continuation<in T> {
    fun resume(value: T)
    fun resumeWithException(exception: Throwable)
}

В новой версии планируется использовать типизированный класс Result, содержащий в себе в каком-то виде значение или ошибку, а в Continuation оставить один метод resume(result: Result<T>):


class Result<out T> {
    val value: T?
    val exception: Throwable?
}

interface Continuation<in T> {
    fun resume(result: Result<T>)
}

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


К сожалению, у такого решения есть проблема. Чтобы все работало в текущей реализации Котлина, необходимо каждое значение оборачивать во wrapper Result, что приводит к созданию лишнего объекта на каждый вызов resume. Решением этой проблемы станут inline классы.


Inline классы


Пример для Result:


inline class Result<out T>(val o: Any?) {
    val value: T?
        get() = if (o is Box) null else o as T
    val exception: Throwable?
        get() = (o as? Box).exception
}

private class Box(val exception: Throwable)

Преимущество inline классов в том, что во время выполнения объект будет храниться просто как val o, переданный в конструкторе, согласно примеру выше, а exception останется в обертке Box. Это обусловлено тем, что ошибки пробрасываются довольно редко, и генерация обертки не повлияет на производительность. Геттеры value и exception будут сгенерированы в статические методы, а сам wrapper исчезает. Это решает проблему с созданием лишних объектов при вызове Continuation.resume.


Ограничения inline классов


  • Пока возможно создать inline класс только с одним полем, так как во время выполнения обертка стирается и остается только значение поля
  • Если используются generics, то Result все равно будет обернут, так как после распаковки неизвестно, обернутый вернулся объект или нет (например, если мы используем List<Result>)

Беззнаковая арифметика


Одно из возможных следствий появления inline классов — unsigned типы:


inline class UInt(val i: Int)
inline class ULong(val l: Long)
...

В стандартной библиотеке для каждого примитива будут описаны соответствующие беззнаковые варианты, в которых будут реализована работа с арифметическими операторами, а также базовые методы, такие как toString().


Также будет добавлена поддержка литералов и неявное преобразование.


Default методы


В интерфейсах Котлина методы со стандартной реализацией появились раньше, чем в Java.
Если интерфейс написан на Котлине, а реализация — на Java 8, то для методов со стандартной реализацией можно будет использовать аннотацию @JvmDefault, чтобы с точки зрения Java этот метод был помечен как default.


Метаинформация для DFA и смарткастов


Ещё одна интересная разрабатываемая штука — контракты, с помощью которых можно добавлять метаинформацию для DFA (Data Flow Analysis) и смарткастов. Рассмотрим пример:


fun test(x: Any) {
    check(x is String)
    println(x.length)
}

В функцию check(Boolean) передается результат проверки на принадлежность x типу String. На данном уровне неизвестно, что эта функция делает внутри, поэтому следующая строка кода вызовет ошибку (smart-cast в данном случае невозможен). Компилятор не может быть уверен, что x это String. Объяснить ему это помогут контракты.


Вот реализация check с контрактом:


fun check(value: Boolean) {
    contract {
        returns() implies value
    }
    if (!value) throw ...
}

Здесь добавлен вызов функции contract с довольно специфическим синтаксисом, который говорит о том, что если этот метод возвращает результат, этот результат точно true. Иначе будет выброшено исключение.


Благодаря такому контракту, компилятор сможет для x выполнить smart-cast, и при вызове println(x.length) не будет ошибки.


Помимо разрешения ситуаций с неизвестными типами, контракты позволят решать и другие проблемы. Например:


fun test() {
    val x: String
    run { x = "Hello" }
    println(x.length)
}

fun <R> run(block: () -> R): R {
    contract {
        callsInPlace(block, EXACTLY_ONCE)
    }
    return block()
}

Контракт в функции run сообщает компилятору, что переданный block будет вызван ровно один раз, константа x корректно проинициализируется и x.length выполнится без ошибки.


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


В IDE планируется определенная подсветка контрактов и автогенерация, где это будет возможно.


Аннотации для управления type inference


В stdlib Котлина уже есть различные фичи, которые используются только внутри. Некоторыми из них разработчики готовы поделиться. Рассмотрим новые аннотации и возможности, которые они предоставляют:


@NoInfer


Аннотация @NoInfer призвана делать type inference немного умнее.


Допустим, мы хотим отфильтровать коллекцию, используя метод filterIsInstance. В данном методе важно указать тип, по которому будет выполняться фильтрация, но компилятор может допустить вызов и без прописывания типа (попытаясь его вывести). Если же в сигнатуре использовать @NoInfer, то вызов без типа выделится, как ошибка.


fun print(c: Collection<Any>) { ... }

val c: Collection<Any>
print(c.filterIsInstance()) //ошибка
print(c.filterIsInstance<String>())

fun <R> Iterable<*>.filterIsInstance(): List<@NoInfer R>

@Exact


Эта аннотация очень похожа на @NoInfer. Она сообщает, что тип должен быть точно равен указанному, то есть не "подтип" и не "надтип".


@OnlyInputTypes


Данная аннотация указывает, что только типы, которые были у пользователя в аргументах или в ресивере, должны быть результатами inference.


SAM для Java методов


Еще одна разрабатываемая возможность упрощает работу с SAM. Рассмотрим пример:


//Java
static void test(Factory f, Runnable r)

interface Factory {
    String produce();
}

//Kotlin
fun use(f: Factory) {
    test(f) { } // Factory != () -> String
}

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


1. test(Factory, Runnable)
2. test(Factory, () -> Unit)
3. test(() -> String, Runnable)
4. test(() -> String, () -> Unit)

Сейчас варианты 2. и 3. невозможны. Компилятор Котлина допускает только два варианта: принимающий два интерфейса и принимающий две лямбды.


Сделать возможными все 4 варианта при текущей реализации компилятора — сложная задача, но не невыполнимая. В новой системе type inference будет поддержка таких ситуаций. В среде будет видна только одна функция fun test(Factory, Runnable), но передавать можно будет как лямбды, так и объекты-реализации интерфейсов в любом сочетании.


SAM для Kotlin методов и интерфейсов


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


sam interface Predicate<T> {
    fun test(t: T): Boolean
}

fun Collection<T>.filter(p: Predicate<T>): Collection<T> { ... }

fun use() {
    test { println("Hello") }
    val l = listOf(-1, 2, 3)
    l.filter { it > 0 }
}

Для Java интерфейсов конверсия будет работать всегда.


Smart inference для билдеров


Представим, что мы хотим написать билдер:


fun <T> buildList(l: MutableList<T>.() -> Unit): List<T> { ... }

И хотим его использовать вот так:


val list = buildList {
    add("one")
    add("two")
}

В данный момент это невозможно, так как тип T не выводится из вызовов add(String). Поэтому приходится писать так:


val list = buildList<String> {
    add("one")
    add("two")
}

Планируется, что можно будет использовать первый вариант (без явного указания типа).


Изменения в схеме компилятора


В JetBrains ведется активная работа над Kotlin Native. Как следствие появилось еще одно звено в схеме компилятора — Back-end Internal Representation (BE IR).



BE IR — это промежуточное представление, содержащее всю семантику исходного кода, который может быть скомпилирован под исполняемые файлы любой платформы, включая бинарный код системы. Сейчас BE IR используется только в Kotlin Native, но его планируется использовать для всех платформ, вместо PSI с дополнительной информацией о семантике. Для JVM и JS уже есть прототипы, и они активно дорабатываются.


В итоге весь исходный код будет преобразовываться в BE IR, а затем в исполняемые файлы целевой платформы.


Резюме


Как было написано выше, неизвестно, какие новшества дойдут до релиза, в каком виде и в какой версии языка. Есть только текущие планы команды Kotlin, но разработчики никаких гарантий не дают:


  • Релиз и финализация API корутин — 1.3
  • Inline классы — экспериментальная возможность в 1.3
  • Беззнаковая арифметика — экспериментальная возможность в 1.3 или 1.4
  • @JvmDefault — экспериментальная возможность в 1.2.x, релиз в 1.3
  • Метаинформация для DFA и смарткастов — частично релиз в 1.3
  • Аннотации для управления type inference — экспериментальная возможность в 1.2.x
  • Новый движок type inference (включая SAM и Smart inference для билдеров) — экспериментальная возможность в 1.3

Kotlin — без сомнения, состоявшийся язык программирования, но с увеличением пользовательской базы расширяется и спектр его применения. Появляются новые требования, платформы и варианты использования. Накапливается старый код, который требуется поддерживать. Разработчики языка понимают сложности и проблемы и работают с ними, в результате новые версии позволят еще удобнее им пользоваться в большем количестве ситуаций.

Комментарии (9)


  1. solver
    19.03.2018 17:27

    Такими темпами, Kotlin обойдет Scala по всем статьям лет за 5.
    Медленно, но верно движутся к светлому будущему. А лайтбенд че то топчется на месте…


    1. guai
      20.03.2018 20:57

      По самому важному свойству — по практичности — имхо, уже. Разрабы скалы долго игрались, затаскивая в язык всё новые модные фичи, а потом внезапно решили всех убедить, что у них получился прагматичный ЯП. Но мало кто убедился :)


  1. lany
    20.03.2018 18:41

    Интересно, а компилятор проверит, что обещания контракта метод действительно выполняет или примет их на веру?


    1. pontifex024 Автор
      20.03.2018 20:00

      Пока неизвестно, как оно будет, но контракт ведь пишет разработчик для конкретного метода. Возможно, выполнение этого контракта тоже будет лежать на совести разработчика.


      1. lany
        22.03.2018 06:57
        +1

        Вот на совести разработчика — это сделает язык дырявым. Простой пример:


        fun test() {
            val x: String
            run { x = "Hello" }
            println(x.length)
        }
        
        fun <R> run(block: () -> R): Unit {
            contract {
                callsInPlace(block, EXACTLY_ONCE)
            }
            //return block() бугагашечка
        }

        В результате, если мы поверили контракту, но не проверили его, то в методе test() мы получаем обращение к неинициализированной локальной переменной — то, чего, например, в Java не может быть никогда и ни за что.


        На самом деле даже если контракты проверяются при компиляции, не забываем, что компиляция раздельная. Если метод run в прекомпилированном классе, и мы там подхачили контракт руками, мы получим тот же эффект. Опасной мне кажется эта фича, в общем. Может, конечно, разработчики умнее меня и как-то всё предусмотрели...


        1. pontifex024 Автор
          22.03.2018 13:27

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


    1. a__v__k
      21.03.2018 21:54

      Для джавы проверки корректности для NotNull и @Contract у Jetbrains есть — внешними средствами.
      Встроить в компилятор котлина аналоги — почему бы и нет.


      1. pontifex024 Автор
        21.03.2018 21:59

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


      1. lany
        22.03.2018 06:53

        Они не стопроцентно работают, их можно обмануть.