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

Kotlin 委托

文章目录

  • 定义
  • 继承/实现的委托
  • 属性的委托
    • 委托给对象
    • 委托给另一个变量
  • 自定义可委托类

定义

有时候我们需要办成某件事但又不想自己做时,可以将其交给别人做,这其实就是一种委托。Kotlin 中可以使用by关键字声明委托。

继承/实现的委托

某个类ClassA想要实现一个接口MyInterface,但是发现接口中的抽象方法非常多,全部实现非常困难。巧的是我们已经有另一个类ClassB(或者该接口的一个对象)实现了该接口,此时我们可以将ClassA的实现MyInterface的工作委托给ClassB

interface MyInterface {
    // 超多抽象方法
    fun example()
}


class ClassB: MyInterface {
    // 超多抽象方法的实现
    override fun example() = print("ClassB 实现")
}


class ClassA(classB: ClassB): MyInterface by classB {
    // ClassA 不需要自己实现 MyInterface 的方法
}


fun main() {
    val classB = ClassB()
    ClassA(classB).example()
}
ClassB 实现

如果我们发现ClassB中的某个方法的实现并不是ClassA想要的,还可以在ClassA中声明对MyInterface方法的重写,此时如果调用ClassAexample则会调用新重写的这个:

interface MyInterface {
    // 超多抽象方法
    fun example()
}


class ClassB: MyInterface {
    // 超多抽象方法的实现
    override fun example() = print("ClassB 实现")
}


class ClassA(classB: ClassB): MyInterface by classB {
    override fun example() = print("ClassA 实现")
}


fun main() {
    val classB = ClassB()
    ClassA(classB).example()
}
ClassA 实现

属性的委托

委托给对象

我们可以将属性 getter 和 setter 委托给某一个对象,此时会将getset函数委托给类的getValuesetValue声明。

val修饰的不可变变量只有 getter,因此只会将get委托给getValue

最常见的是Lazy,我们可以使用函数lazy(后边的 lambda 需要返回一个值作为变量的初始值,该值的类型会作为变量类型)生成一个Lazy对象,并将某一个变量委托给它:

val name = "Kotlin".also { println("name 初始化") }
val lazyName by lazy { println("lazyName 初始化"); "Kotlin" }

val version = 2.also { println("version 初始化") }
val lazyVersion by lazy { println("lazyVersion 初始化"); 2 }

fun main() {
    // 在运行时,name 就被初始化了
    // 而委托给 Lazy 的 lazyName 没有被初始化

    // 在运行时,version 就被初始化了
    // 只有在访问值时,lazyVersion 才会被初始化
    lazyVersion
}
name 初始化
version 初始化
lazyVersion 初始化

委托给另一个变量

此时委托变量前需要加双冒号::,事实上,::name会返回一个KMutableProperty0<String>对象,所以这是还是将变量委托给了对象。比较特殊的是,该对象并没有声明getValuesetValue,委托将交给其 getter 和 setter。

二者的 getter 和 setter 实际上已经绑定在一起了,当其中一个的值改变,另一个也会跟着变。

var name: String = "K1"


fun main() {
    var delegateName by ::name
    
    // 打印 name 并改变 delegateName 也是同样的结果
    println(delegateName)
    name = "K2"
    print(delegateName)
}
K1
K2

能将可变变量var委托给不可变变量val,因为val没有 setter。

val name: String = "K1"


fun main() {
    // 这是错误的
    // var delegateName by ::name
}

自定义可委托类

得益于 IDEA 自动补全,我们可以直接写出委托关系var value by MyClass(),鼠标悬停(或者光标置于标红处,按键盘Alt+Enter),点击创建getValuesetValue,并加以修改即可。其中,参数名可以自定义。

生成的nothing(第1个参数)一般是叫thisRef,它的类型是该变量所有者的类型(例如某个类的成员变量,其所有者是该类)。
这里Nothing?则表示没有所有者。对于其他类型,如果加了?,则所有变量都可以委托,如果不加,则只有变量的所有者为指定类型可以委托。

property中则包含了委托变量的属性,例如property.name可以获取到变量名。其类型必须为KProperty<*>

getValue的返回值一般需要指明为所要获取值的类型,这里把MyClass.value返回。
setValue传入的第3个参数则是要赋的值,这里把值给MyClass.value

import kotlin.reflect.KProperty


class MyClass {
    private var value = "MyClass"


    operator fun getValue(thisRef: Nothing?, property: KProperty<*>): String {
        println("getValue")
        return value
    }


    operator fun setValue(thisRef: Nothing?, property: KProperty<*>, s: String) {
        println("setValue")
        value = s
    }
}


fun main() {
    var value by MyClass()
    println(value)
    value = "Hello"
    print(value)
}
getValue
MyClass
setValue
getValue
Hello

相关文章:

  • 面试Flask需要知道的知识点1
  • 【目标检测】基于深度学习的车牌识别管理系统(含UI界面)【python源码+Pyqt5界面 MX_002期】
  • 【Nginx系列】基于请求头的分发
  • Linux fallocate工具用于预分配或释放文件空间的块
  • 【qt】项目移植
  • pdf添加书签的软件,分享3个实用的软件!
  • LLVM Cpu0 新后端6
  • 【HarmonyOS】HUAWEI DevEco Studio 下载地址汇总
  • SQL学习小记(六)将数据库中表为NULL的地方补上需要的值
  • NSSCTF中的popchains、level-up、 What is Web、 Interesting_http、 BabyUpload
  • 防火墙安全管理
  • 【数据分析基础】实验一 Python运算符、内置函数、序列基本用法
  • RabbitMQ 见解一
  • go语言后端开发学习(一)——JWT的介绍以及基于JWT实现登录验证
  • 办理水文设施设计乙级资质如何对申报材料进行保护
  • Vue 路由传递参数 query、params
  • 鸿蒙轻内核M核源码分析系列二十 Newlib C
  • 文化融合,市场共赢:品牌海外推广中的符号与象征策略
  • Elixir学习笔记——二进制、字符串和字符列表
  • 《计算机组成原理》期末复习题节选
  • 世界哮喘日|专家:哮喘无法根治,“临床治愈”已成治疗新目标
  • 上海市政府常务会议部署提升入境旅游公共服务水平,让国际友人“无障碍”畅游上海
  • 哈马斯官员:进一步停火谈判毫无意义
  • 美法官裁定特朗普援引战时法律驱逐黑帮违法,系首次永久性驳回
  • 山西太原一小区发生爆炸,太原:进一步深刻汲取教训
  • 韩国法院将重审李在明案,韩德洙入局能否为大选带来变数?