Делегирование
Шаблон делегирования является хорошей альтернативой наследованию, и Kotlin поддерживает его нативно, освобождая вас от необходимости написания шаблонного кода.
Класс Derived
может реализовать интерфейс Base
, делегируя все свои public члены указанному объекту.
interface Base {
fun print()
}
class BaseImpl(val x: Int) : Base {
override fun print() { print(x) }
}
class Derived(b: Base) : Base by b
fun main() {
val b = BaseImpl(10)
Derived(b).print()
}
Ключевое слово by
в оглавлении Derived
, указывает, что b
будет храниться внутри экземпляра Derived
, и компилятор
сгенерирует все методы из Base
, которые при вызове будут переданы объекту b
.
Переопределение члена интерфейса, реализованного делегированием
Переопределения работают так, как вы ожидаете: компилятор будет использовать
ваши override
реализации вместо реализаций в объекте делегата. Если вы хотите добавить
override fun printMessage() { print("abc") }
в Derived
, программа будет печатать abc вместо 10 при вызове printMessage
.
interface Base {
fun printMessage()
fun printMessageLine()
}
class BaseImpl(val x: Int) : Base {
override fun printMessage() { print(x) }
override fun printMessageLine() { println(x) }
}
class Derived(b: Base) : Base by b {
override fun printMessage() { print("abc") }
}
fun main() {
val b = BaseImpl(10)
Derived(b).printMessage()
Derived(b).printMessageLine()
}
Однако обратите внимание, что члены, переопределенные таким образом, не вызываются из членов делегата, который может получить доступ только к своим собственным реализациям членов интерфейса.
interface Base {
val message: String
fun print()
}
class BaseImpl(val x: Int) : Base {
override val message = "BaseImpl: x = $x"
override fun print() { println(message) }
}
class Derived(b: Base) : Base by b {
// Это свойство недоступно из `b` реализации `print`
override val message = "Message of Derived"
}
fun main() {
val b = BaseImpl(10)
val derived = Derived(b)
derived.print()
println(derived.message)
}
Узнайте больше о делегированных свойствах.