Edit Page

委譲 (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 節は、 bDerived のオブジェクトに内部的に格納されることを示し、コンパイラは 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) }

委譲プロパティでさらに詳しく学ぼう。