委托

属性委托

属性委托在单独一页中讲:属性委托

由委托实现

委托模式已经证明是实现继承的一个很好的替代方式, 而 Kotlin 可以零样板代码地原生支持它。 Derived 类可以通过将其所有公有成员都委托给指定对象来实现一个接口 Base

  1. interface Base {
  2. fun print()
  3. }
  4. class BaseImpl(val x: Int) : Base {
  5. override fun print() { print(x) }
  6. }
  7. class Derived(b: Base) : Base by b
  8. fun main() {
  9. val b = BaseImpl(10)
  10. Derived(b).print()
  11. }

Derived 的超类型列表中的 by-子句表示 b 将会在 Derived 中内部存储, 并且编译器将生成转发给 b 的所有 Base 的方法。

覆盖由委托实现的接口成员

覆盖符合预期:编译器会使用 override 覆盖的实现而不是委托对象中的。如果将 override fun printMessage() { print("abc") } 添加到 Derived,那么当调用 printMessage 时程序会输出“abc”而不是“10”:

  1. interface Base {
  2. fun printMessage()
  3. fun printMessageLine()
  4. }
  5. class BaseImpl(val x: Int) : Base {
  6. override fun printMessage() { print(x) }
  7. override fun printMessageLine() { println(x) }
  8. }
  9. class Derived(b: Base) : Base by b {
  10. override fun printMessage() { print("abc") }
  11. }
  12. fun main() {
  13. val b = BaseImpl(10)
  14. Derived(b).printMessage()
  15. Derived(b).printMessageLine()
  16. }

但请注意,以这种方式重写的成员不会在委托对象的成员中调用 ,委托对象的成员只能访问其自身对接口成员实现:

  1. interface Base {
  2. val message: String
  3. fun print()
  4. }
  5. class BaseImpl(val x: Int) : Base {
  6. override val message = "BaseImpl: x = $x"
  7. override fun print() { println(message) }
  8. }
  9. class Derived(b: Base) : Base by b {
  10. // 在 b 的 `print` 实现中不会访问到这个属性
  11. override val message = "Message of Derived"
  12. }
  13. fun main() {
  14. val b = BaseImpl(10)
  15. val derived = Derived(b)
  16. derived.print()
  17. println(derived.message)
  18. }