当前位置: 首页 > web >正文

Kotlin委托

委托(Delegation)

大多数编程工作都涉及重用已有代码,有时只做少量改动。在面向对象编程中(Kotlin 也是如此),代码复用的主要手段是继承(以及组合),我们之前已经讲过了。这节内容我们讲另一种替代继承的方式——委托


委托的语法

委托是使用某个对象来完成接口的实现,而不是在当前类里重新写一遍实现。我们来看看具体怎么做。

假设我们有这样一段简单的代码——一个接口和它的实现:

interface MyInterface {fun print()val msg: String
}class MyImplementation : MyInterface {override fun print() {println(msg)}override val msg: String = "MyImplementation sends regards!"
}

解释:
接口声明了一个属性和一个方法,类 MyImplementation 实现了它们。


现在,假设我们想创建一个新类,这个类要:

  1. 拥有自己的功能,

  2. 同时实现上述接口。

如果直接用继承,可能需要复制粘贴已有代码;但用委托就能避免这种重复。


用委托写法:

class MyNewClass(base: MyInterface) : MyInterface by base {override val msg = "Delegate sends regards."
}

解释:

  • 构造函数参数 base 期望传入一个 MyInterface 的实现。

  • 冒号后面写 MyInterface by base,表示 MyNewClass 实现了接口 MyInterface,但接口的具体实现“委托”给了 base

  • msg 属性被重写成了新的值。


实例代码:

val delegate = MyImplementation()
val delegatingObj = MyNewClass(delegate)
println(delegatingObj.msg)

输出:

Delegate sends regards.

但是,如果调用 print() 呢?

delegatingObj.print()

它会打印:

MyImplementation sends regards!

解释:
虽然 MyNewClass 没有自己写 print() 方法,但它委托给了 base,即 MyImplementation 的实例,调用的就是 MyImplementation 里的 print()。而 print() 内部打印的是 MyImplementationmsg,所以输出是 MyImplementation sends regards!

总结:

  • MyNewClass 中重写的成员会被使用。

  • 没有重写的方法和属性则直接委托给 base 实现。


一个更复杂的例子 — 回调和日志器

现在,我们看一个包含两个委托的复杂例子。

  • ICallbackReceiver:回调接口,支持在执行一个动作前后调用特定函数。

  • ILogger:日志接口,负责格式化并输出日志。

// 回调接口
interface ICallbackReceiver {fun onBeforeAction()fun onAfterAction()fun action(function: () -> Unit) {onBeforeAction()function()onAfterAction()}
}// 日志接口
interface ILogger {fun getStubDateTime() = "05.11.2022-14:31:04" // 占位时间val format: Stringget() = "[${getStubDateTime()}]: "fun print(s: String)
}

实现:

// 简单的日志实现
class BasicLogger : ILogger {override fun print(s: String) = println(format + s)
}// 实现回调接口,日志功能委托给 logger
class ConsoleNotifier(logger: ILogger) : ICallbackReceiver, ILogger by logger {val onBeforeStr = "OnBefore!"val onAfterStr = "OnAfter!"override fun onBeforeAction() = print(onBeforeStr)override fun onAfterAction() = print(onAfterStr)
}

再定义一个既支持回调又支持日志的类,全部用委托完成:

class ExampleParser(notifier: ICallbackReceiver, logger: ILogger) :ICallbackReceiver by notifier,ILogger by logger {fun start() = action { parseFiles() }fun parseFiles() {print("Parsing...")// 这里写具体的解析逻辑}
}

运行示例:

fun main() {val loggerInstance = BasicLogger()val dateTimeNotifier = ConsoleNotifier(loggerInstance)val simpleParser = ExampleParser(dateTimeNotifier, loggerInstance)simpleParser.start()
}

输出:

[05.11.2022-14:31:04]: OnBefore!
[05.11.2022-14:31:04]: Parsing...
[05.11.2022-14:31:04]: OnAfter!

总结

委托极大提升了代码复用的便利性。相比复制粘贴已有代码,我们只需引入已有的功能对象,并把接口的实现委托给它即可。Kotlin 对委托的语言支持非常好,让这项工作简单且优雅。

http://www.xdnf.cn/news/15495.html

相关文章:

  • python的小学课外综合管理系统
  • MFC UI大小改变与自适应
  • ICMP隧道工具完全指南:原理、实战与防御策略
  • SwiftUI 全面介绍与使用指南
  • 【Docker基础】Dockerfile指令速览:健康检查与启动指令详解
  • Spark SQL 之 UT
  • Linux操作系统从入门到实战(七)详细讲解编辑器Vim
  • C++-多态
  • 现场设备无法向视频汇聚EasyCVR视频融合平台推流的原因排查与解决过程
  • Java Integer包装类缓存机制详解
  • python中正则相关:正则概述,匹配数字,匹配单词,匹配字符开头与结尾,单词的边界检测
  • MySQL数据实时同步到Elasticsearch的高效解决方案
  • aspnetcore Mvc配置选项中的ModelMetadataDetailsProviders
  • Ubuntu服务器安装Miniconda
  • 鸿蒙应用开发: 鸿蒙项目中使用私有 npm 插件的完整流程
  • 华为MateBook D 16 SE版 2024款 12代酷睿版i5集显(MCLF-XX,MCLF-16)原厂OEM预装Win11系统
  • vscode 打开项目时候,有部分外部依赖包找不到定义或者声明,但是能使用cmake正常编译并且运行
  • 【前端】【Iconify图标库】【vben3】createIconifyIcon 实现图标组件的自动封装
  • AWS RDS PostgreSQL可观测性最佳实践
  • Linux操作系统从入门到实战(八)详细讲解编译器gcc/g++编译步骤与动静态库链接
  • S7-1200 中 AT 覆盖参数的应用:灵活访问数据区域的实用指南
  • 银河麒麟KYSEC安全机制详解
  • Java设计模式(java design patterns)
  • 【linux V0.11】boot
  • 【算法深练】BFS:“由近及远”的遍历艺术,广度优先算法题型全解析
  • 掉线监测-tezos rpc不能用,改为残疾网页监测
  • 视频孪生与三维融合:智汇云舟赋能智慧化电力转型的破局之道
  • 【数据结构初阶】--单链表(二)
  • Flask服务器公外网访问,IPv6(亲测有效!!!!)
  • 哈希扩展 --- 海量数据处理