Skip to content

委托

委托模式 (Delegation pattern) 已被证明是实现继承 (implementation inheritance) 的一个良好替代方案,而 Kotlin 原生支持它,无需样板代码 (boilerplate code)。

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 对象中,并且编译器将生成所有转发到 bBase 方法。

覆盖通过委托实现的接口成员

覆盖 (Overrides) 按预期工作:编译器将使用你的 override 实现,而不是委托对象中的实现。如果你想向 Derived 添加 override fun printMessage() { print("abc") },当调用 printMessage 时,程序将打印 abc 而不是 10

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

了解更多关于委托属性 (delegated properties)