委譲 (Delegation)
Delegationパターンは、実装継承の良い代替手段であることが証明されており、Kotlinはネイティブでそれをサポートし、かつ定型コードを必要としません。
Derived
クラスは、
指定されたオブジェクトへ public メンバのすべてを委譲することで、
Base
インターフェースを実装する事が出来ます:
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()
}
Derived
の基底型のリスト中の by 節は、 b
が Derived
のオブジェクトに内部的に格納されることを示し、コンパイラは Base
のすべてのメソッドをb
に取り次ぐものとして生成します。
委譲で実装されるインターフェースのoverride
オーバーライドは期待される通りの振る舞いをします:
コンパイラはあなたの作ったoverride
の実装の方をdelegateオブジェクトの物の代わりに使います。
Derived
クラスにoverride fun printMessage() { print("abc") }
を足せば、
プログラムはprintMessage
が呼ばれた時に10の代わりにabcを出力するようになります:
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()
}
しかしながら、delegateオブジェクトからはこのようにオーバーライドしたメンバは呼ばれない事には注意が必要です。 delegateオブジェクトは自身の実装にしかアクセス出来ないのですから:
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 = "Derivedのメッセージ"
}
fun main() {
val b = BaseImpl(10)
val derived = Derived(b)
derived.print()
println(derived.message)
}
委譲プロパティでさらに詳しく学ぼう。