メインコンテンツまでスキップ

委譲

Delegation patternは実装の継承の良い代替手段であることが証明されており、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 base = BaseImpl(10)
Derived(base).print()
}

Derivedのスーパータイプリストのby-clauseは、bDerivedのオブジェクト内に内部的に格納され、コンパイラーがbに転送するBaseのすべてのメソッドを生成することを示します。

委譲によって実装されたインターフェースのメンバーのオーバーライド

Overridesは期待どおりに機能します。コンパイラーは、デリゲートオブジェクトのメソッドの代わりに、記述したoverride実装を使用します。override fun printMessage() { print("abc") }Derivedに追加すると、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 base = BaseImpl(10)
Derived(base).printMessage()
Derived(base).printMessageLine()
}

ただし、この方法でオーバーライドされたメンバーは、デリゲートオブジェクトのメンバーからは呼び出されません。デリゲートオブジェクトは、インターフェースメンバーの独自の実装にのみアクセスできます。

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)
}

詳細については、delegated propertiesを参照してください。