Enum-классы
Наиболее базовый пример использования enum (перечисления) — это реализация типобезопасных перечислений.
enum class Direction {
NORTH, SOUTH, WEST, EAST
}
Каждая enum-константа является объектом. При объявлении константы разделяются запятыми.
Так как каждая enum-константа является экземпляром enum-класса, она может быть инициализирована:
enum class Color(val rgb: Int) {
RED(0xFF0000),
GREEN(0x00FF00),
BLUE(0x0000FF)
}
Анонимные классы
Enum-константы могут объявлять собственные анонимные классы с соответствующими методами, а также переопределять методы базового класса.
enum class ProtocolState {
WAITING {
override fun signal() = TALKING
},
TALKING {
override fun signal() = WAITING
};
abstract fun signal(): ProtocolState
}
Если enum-класс определяет какие-либо члены, отделяйте объявления констант от объявлений членов точкой с запятой.
Реализация интерфейсов в классах enum
Enum-класс может реализовывать интерфейс (но не наследоваться от класса), предоставляя либо общую реализацию членов интерфейса для всех записей enum, либо отдельные реализации для каждой записи в её анонимном классе. Для этого добавьте интерфейсы, которые нужно реализовать, в объявление enum-класса:
import java.util.function.BinaryOperator
import java.util.function.IntBinaryOperator
enum class IntArithmetics : BinaryOperator<Int>, IntBinaryOperator {
PLUS {
override fun apply(t: Int, u: Int): Int = t + u
},
TIMES {
override fun apply(t: Int, u: Int): Int = t * u
};
override fun applyAsInt(t: Int, u: Int) = apply(t, u)
}
fun main() {
val a = 13
val b = 31
for (f in IntArithmetics.entries) {
println("$f($a, $b) = ${f.apply(a, b)}")
}
}
Все enum-классы по умолчанию реализуют интерфейс Comparable. Константы enum-класса объявляются в естественном порядке. Подробнее см. в разделе Сортировка.
Работа с enum-константами
Enum-классы в Kotlin имеют синтетические свойства и методы для получения списка объявленных enum-констант и получения
enum-константы по её имени. Ниже приведены сигнатуры этих методов (при условии, что имя enum-класса — EnumClass):
EnumClass.valueOf(value: String): EnumClass
EnumClass.entries: EnumEntries<EnumClass> // специализированный List<EnumClass>
Ниже приведён пример их использования:
enum class RGB { RED, GREEN, BLUE }
fun main() {
for (color in RGB.entries) println(color.toString()) // выведет RED, GREEN, BLUE
println("The first color is: ${RGB.valueOf("RED")}") // выведет "The first color is: RED"
}
Метод valueOf() выбрасывает исключение IllegalArgumentException, если указанное имя не соответствует ни одной
enum-константе, объявленной в классе.
До появления entries в Kotlin 1.9.0 для получения массива enum-констант использовалась функция values().
У каждой enum-константы также есть свойства name
и ordinal, которые позволяют получить её имя
и позицию (начиная с 0) в объявлении enum-класса:
enum class RGB { RED, GREEN, BLUE }
fun main() {
println(RGB.RED.name) // выведет RED
println(RGB.RED.ordinal) // выведет 0
}
Совет. Чтобы сократить повторения при работе с записями enum, попробуйте контекстно-зависимое разрешение (сейчас доступно в предварительном режиме). Эта возможность позволяет опускать имя enum-класса, когда ожидаемый тип известен, например в выражениях
whenили при присваивании переменной с явным типом.Подробнее см. в разделе Preview of context-sensitive resolution и в соответствующем KEEP-предложении.
К константам в enum-классе можно получить доступ универсальным способом, используя функции
enumValues<T>() и
enumValueOf<T>(). В Kotlin 2.0.0 функция
enumEntries<T>() появилась как замена
функции enumValues<T>(). Функция
enumEntries<T>() возвращает список всех записей enum для заданного enum-типа T.
Функция enumValues<T>() по-прежнему поддерживается, но рекомендуется использовать вместо неё enumEntries<T>(),
поскольку она оказывает меньшее влияние на производительность. При каждом вызове enumValues<T>() создаётся новый
массив, тогда как enumEntries<T>() каждый раз возвращает один и тот же список, что намного эффективнее.
Например:
enum class RGB { RED, GREEN, BLUE }
inline fun <reified T : Enum<T>> printAllValues() {
println(enumEntries<T>().joinToString { it.name })
}
printAllValues<RGB>()
// RED, GREEN, BLUE
Подробнее о встроенных функциях и reified-параметрах типа см. в разделе Встроенные (inline) функции.