デリゲーション
デリゲーションパターンは、実装継承に代わる優れた方法であることが証明されており、Kotlinはボイラープレートコードを一切必要とせずにこれをネイティブにサポートしています。
Derived
クラスは、すべてのパブリックメンバーを指定されたオブジェクトにデリゲートすることで、Base
インターフェースを実装できます。
kotlin
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 base = BaseImpl(10)
Derived(base).print()
}
Derived
のスーパタイプリストにあるby
句は、b
がDerived
のオブジェクト内部に格納され、コンパイラがb
に転送するBase
のすべてのメソッドを生成することを示しています。
デリゲーションによって実装されたインターフェースのメンバーをオーバーライドする
オーバーライドは期待どおりに機能します。コンパイラはデリゲートオブジェクト内の実装ではなく、あなたのoverride
実装を使用します。Derived
にoverride fun printMessage() { print("abc") }
を追加すると、printMessage
が呼び出されたときにプログラムは10の代わりにabcを出力します。
kotlin
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 base = BaseImpl(10)
Derived(base).printMessage()
Derived(base).printMessageLine()
}
ただし、この方法でオーバーライドされたメンバーは、デリゲートオブジェクトのメンバーからは呼び出されないことに注意してください。デリゲートオブジェクトは、インターフェースメンバーの自身の実装にのみアクセスできるためです。
kotlin
interface Base {
val message: String
fun print()
}
class BaseImpl(x: Int) : Base {
override val message = "BaseImpl: x = $x"
override fun print() { println(message) }
}
class Derived(b: Base) : Base by b {
// This property is not accessed from b's implementation of `print`
override val message = "Message of Derived"
}
fun main() {
val b = BaseImpl(10)
val derived = Derived(b)
derived.print()
println(derived.message)
}
デリゲートプロパティの詳細については、こちらをご覧ください。