委托
属性委托
属性委托在单独一页中讲:属性委托。
由委托实现
委托模式已经证明是实现继承的一个很好的替代方式, 而 Kotlin 可以零样板代码地原生支持它。 Derived 类可以通过将其所有公有成员都委托给指定对象来实现一个接口 Base:
interfaceBase{
funprint()
}
classBaseImpl(valx:Int) :Base{
overridefunprint() {print(x) }
}
classDerived(b:Base) :Basebyb
funmain() {
valb=BaseImpl(10)
Derived(b).print()
}
Target platform: JVMRunning on kotlin v. 1.3.70
Derived 的超类型列表中的 by-子句表示 b 将会在 Derived 中内部存储, 并且编译器将生成转发给 b 的所有 Base 的方法。
覆盖由委托实现的接口成员
覆盖符合预期:编译器会使用 override 覆盖的实现而不是委托对象中的。如果将 override fun printMessage() { print("abc") } 添加到 Derived,那么当调用 printMessage 时程序会输出“abc”而不是“10”:
interfaceBase{
funprintMessage()
funprintMessageLine()
}
classBaseImpl(valx:Int) :Base{
overridefunprintMessage() {print(x) }
overridefunprintMessageLine() {println(x) }
}
classDerived(b:Base) :Basebyb{
overridefunprintMessage() {print("abc") }
}
funmain() {
valb=BaseImpl(10)
Derived(b).printMessage()
Derived(b).printMessageLine()
}
Target platform: JVMRunning on kotlin v. 1.3.70
但请注意,以这种方式重写的成员不会在委托对象的成员中调用 ,委托对象的成员只能访问其自身对接口成员实现:
interfaceBase{
valmessage:String
funprint()
}
classBaseImpl(valx:Int) :Base{
overridevalmessage="BaseImpl: x = $x"
overridefunprint() {println(message) }
}
classDerived(b:Base) :Basebyb{
// 在 b 的 `print` 实现中不会访问到这个属性
overridevalmessage="Message of Derived"
}
funmain() {
valb=BaseImpl(10)
valderived=Derived(b)
derived.print()
println(derived.message)
}