Skip to content

デリゲーション

デリゲーションパターンは、実装継承に代わる優れた方法であることが証明されており、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句は、bDerivedのオブジェクト内部に格納され、コンパイラがbに転送するBaseのすべてのメソッドを生成することを示しています。

デリゲーションによって実装されたインターフェースのメンバーをオーバーライドする

オーバーライドは期待どおりに機能します。コンパイラはデリゲートオブジェクト内の実装ではなく、あなたのoverride実装を使用します。Derivedoverride 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)
}

デリゲートプロパティの詳細については、こちらをご覧ください。