Kotlin教程笔记(17) - 属性代理

本文涉及的产品
传统型负载均衡 CLB,每月750个小时 15LCU
应用型负载均衡 ALB,每月750个小时 15LCU
公网NAT网关,每月750个小时 15CU
简介: Kotlin教程笔记(17) - 属性代理

本系列学习教程笔记属于详细讲解Kotlin语法的教程,需要快速学习Kotlin语法的小伙伴可以查看“简洁” 系列的教程

快速入门请阅读如下简洁教程:
Kotlin学习教程(一)
Kotlin学习教程(二)
Kotlin学习教程(三)
Kotlin学习教程(四)
Kotlin学习教程(五)
Kotlin学习教程(六)
Kotlin学习教程(七)
Kotlin学习教程(八)
Kotlin学习教程(九)
Kotlin学习教程(十)

Kotlin教程笔记(17) - 属性代理

imgKotlin - 属性代理

自定义属性代理

代理也叫委托,Kotlin 支持属性代理,把一个属性的获取与赋值交给一个“中介”(下称 Delegate)去管理,属性代理的语法是: val/var <属性名>: <类型> by <表达式> ,by 后面是 Delegate 对象,被代理属性的 get()和 set() 会给 Delegate 对象对应的 getValue() 和 setValue() 分别代理,因此 Delegete 可以这么写:

注意:Delegate 并不需要实现任何接口,仅需提供 getValue() 和 setValue() 即可。

class Delegate {
private var _value: String? = null

operator fun getValue(thisRef: Any?, property: KProperty<*>): String {
    println("getValue() thisRef=$thisRef, property=${property.name}")
    return _value ?: ""
}

operator fun setValue(thisRef: Any?, property: KProperty<*>, value: String) {
    println("setValue() thisRef=$thisRef, property=${property.name}, value=$value")
    _value = value
}

}
该 Delegate 的 getValue() 和 setValue() 方法对自己的 _value 属性进行获取与赋值,并分别在这 2 个方法中打印输出各参数值,下面我们将一个类成员属性交给 Delegate 对象代理:

class Animal {
var name: String by Delegate()
}
fun main() {
val animal = Animal()
animal.name = "旺财" // setValue() thisRef=com.charylin.kotlinlearn.Animal@1b28cdfa, property=name, value=旺财
println(animal.name) // getValue() thisRef=com.charylin.kotlinlearn.Animal@1b28cdfa, property=name
   // 旺财
}
当代理属性被赋值与访问时,就会输出 Delegate 对象 setValue() 和 getValue() 中的日志,从输出的日志可以看出,thisRef 是代理属性的实例对象,property 是代理属性的包装。可见,属性代理相比单纯的属性操作更加强大,在某些场景下,属性值的访问会比较复杂,如文件或 Redis,属性代理可以将文件或 Redis 的操作全部交给 Delegate 完成,而业务代码不需要知道具体实现,这对代码解耦与简化很有帮助。

延迟属性 lazy

Kotlin 官方提供了 lazy 代理,可以延迟初始化 val 常量,且只会初始化一次:

class Animal {
val age: Int by lazy {
println("设置age")
18 // lambda表达式会把最后语句的执行结果作为返回值
}
}
fun main() {
val animal = Animal()
println(animal.age) // 设置age
// 18
println(animal.age) // 18
}
可以看到 println("设置age") 只被输出了一次,这是怎么办到的呢?按住 ctrl +鼠标左键,点击 lazy 查看源代码:

// LazyJVM.kt
public actual fun lazy(initializer: () -> T): Lazy = SynchronizedLazyImpl(initializer)
lazy() 函数需要传入一个 lambda 表达式,返回值是 Lazy,返回结果 SynchronizedLazyImpl(initializer) 就是 Lazy 的实现类对象,我们知道 by 后面接的是 Delegate 对象,lazy() 函数返回的 SynchronizedLazyImpl 就是真正的 Delegate ,再来看看 Lazy 与 SynchronizedLazyImpl 的源代码:

// Lazy.kt
public interface Lazy {
public val value: T
...
}

// LazyJVM.kt
private class SynchronizedLazyImpl(initializer: () -> T, lock: Any? = null) : Lazy, Serializable {
private var initializer: (() -> T)? = initializer
@Volatile private var _value: Any? = UNINITIALIZED_VALUE
...

override val value: T
    get() {
        val _v1 = _value
        if (_v1 !== UNINITIALIZED_VALUE) {
            // 已经初始化过了,直接返回
            return _v1 as T
        }

        return synchronized(lock) {
            ...
            // 执行initializer()初始化_value值
            val typedValue = initializer!!()
            _value = typedValue
            initializer = null
        }
    }
...

}
SynchronizedLazyImpl 重写了 value 属性的 getter 方法,其中会判断 value 是否有初始化过,已经初始化就直接返回结果,未初始化才执行 initializer,所以,这就是 by lazy{} 的初始化逻辑只会执行一次的原因。前面说过,Delegate 对象需要有 getValue() 和 setValue() 方法,那么 Lazy 的这 2 个方法在哪呢?按住 ctrl +鼠标左键,点击 by 可以找到:

// Lazy.kt
@kotlin.internal.InlineOnly
public inline operator fun Lazy.getValue(thisRef: Any?, property: KProperty<*>): T = value
发现,Lazy 只有 getValue(),没有 setValue(),这不难理解,var 变量延迟初始化使用的是 lateinit,val 常量使用的 by lazy,val 常量不可修改,所以不需要提供 setValue()。

可观察属性 Observable

除了 by lazy 外,Kotlin 为还提供了可观察属性 Observable,这个属性代理可以监听属性值的变化,需要通过 by Delegates.observable() 来设置,这是 Delegates.observable() 的源码,具体实现就不深入了,该函数要求传入 2 个参数,分别是初始值 initialValue 和用于监听值变化的 lambda 表达式:

// Delegates.kt
public object Delegates {
/**

* Returns a property delegate for a read/write property that calls a specified callback function when changed.
* @param initialValue the initial value of the property.
* @param onChange the callback which is called after the change of the property is made. The value of the property
*  has already been changed when this callback is invoked.
*
*  @sample samples.properties.Delegates.observableDelegate
*/
public inline fun <T> observable(initialValue: T, crossinline onChange: (property: KProperty<*>, oldValue: T, newValue: T) -> Unit):
        ReadWriteProperty<Any?, T> =
    object : ObservableProperty<T>(initialValue) {
        override fun afterChange(property: KProperty<*>, oldValue: T, newValue: T) = onChange(property, oldValue, newValue)
    }
...

}
下面我们来使用 Delegates.observable() 监听属性值的变化:

class Animal {
var gender: Boolean by Delegates.observable(true) { prop, old, new ->
println("prop = ${prop.name}, old = $old, new = $new")
}
}

fun main() {
val animal = Animal()
println(animal.gender) // true
animal.gender = false // prop = gender, old = true, new = false
println(animal.gender) // false
}
可以看到,当修改了属性值时,会触发 Delegates.observable() 参数 2 的 lambda 表达式,输出了相应的日志信息,随后属性值也发生了改变。Kotlin 还提供了 Delegates.vetoable() 可观察属性代理,vetoable 比 observable 要强大一点,它除了能监听属性值变化,还可以控制属性是否要修改,通过查看 Delegates.vetoable() 源码注释可以知道,其参数 2 的 lambda 表达式(也就是 callback)返回值将决定属性值是否被更新:

// Delegates.kt
public object Delegates {
/**

 * Returns a property delegate for a read/write property that calls a specified callback function when changed,
 * allowing the callback to veto the modification.
 * @param initialValue the initial value of the property.
 * @param onChange the callback which is called before a change to the property value is attempted.
 *  The value of the property hasn't been changed yet, when this callback is invoked.
 *  If the callback returns `true` the value of the property is being set to the new value,
 *  and if the callback returns `false` the new value is discarded and the property remains its old value.
 *
 *  @sample samples.properties.Delegates.vetoableDelegate
 *  @sample samples.properties.Delegates.throwVetoableDelegate
 */
public inline fun <T> vetoable(initialValue: T, crossinline onChange: (property: KProperty<*>, oldValue: T, newValue: T) -> Boolean):
        ReadWriteProperty<Any?, T> =
    object : ObservableProperty<T>(initialValue) {
        override fun beforeChange(property: KProperty<*>, oldValue: T, newValue: T): Boolean = onChange(property, oldValue, newValue)
    }
...

}
我们将上面例子中的 gender 属性代理修改成 Delegates.vetoable(),并在 callback 中返回 false,以阻止属性值的修改:

class Animal {
var gender: Boolean by Delegates.vetoable(true) { prop, old, new ->
println("prop = ${prop.name}, old = $old, new = $new")
false // lambda表达式最后语句的结果作为返回值
}
}

fun main() {
val animal = Animal()
println(animal.gender) // true
animal.gender = false // prop = gender, old = true, new = false
println(animal.gender) // true
}
至此属性代理的原理及常用属性代理就讲解完了,如需了解更多属性代理的应用,请点击:https://www.kotlincn.net/docs/reference/delegated-properties.html

相关文章
|
JavaScript 前端开发 索引
如何给组件的元素添加事件监听器?
在组件的元素上添加事件监听器,可以通过在元素上使用 `@事件名` 的语法来实现。例如,`@click=&quot;handleClick&quot;` 表示当元素被点击时,会触发 `handleClick` 方法。
|
6月前
|
机器学习/深度学习 JSON 算法
京东拍立淘图片搜索 API 接入实践:从图像识别到商品匹配的技术实现
京东拍立淘图片搜索 API 是基于先进图像识别技术的购物搜索接口,支持通过上传图片、URL 或拍摄实物搜索相似商品。它利用机器学习和大数据分析,精准匹配商品特征,提供高效、便捷的搜索体验。接口覆盖京东海量商品资源,不仅支持外观、颜色等多维度比对,还结合用户行为数据实现智能推荐。请求参数包括图片 URL 或 Base64 编码,返回 JSON 格式的商品信息,如 ID、价格、链接等,助力消费者快速找到心仪商品,满足个性化需求。
473 18
|
运维 供应链 安全
SD-WAN分布式组网:构建高效、灵活的企业网络架构
本文介绍了SD-WAN(软件定义广域网)在企业分布式组网中的应用,强调其智能化流量管理、简化的网络部署、弹性扩展能力和增强的安全性等核心优势,以及在跨国企业、多云环境、零售连锁和制造业中的典型应用场景。通过合理设计网络架构、选择合适的网络连接类型、优化应用流量优先级和定期评估网络性能等最佳实践,SD-WAN助力企业实现高效、稳定的业务连接,加速数字化转型。
SD-WAN分布式组网:构建高效、灵活的企业网络架构
|
6月前
|
数据采集 运维 API
把Postman调试脚本秒变Python采集代码的三大技巧
本文介绍了如何借助 Postman 调试工具快速生成 Python 爬虫代码,并结合爬虫代理实现高效数据采集。文章通过“跨界混搭”结构,先讲解 Postman 的 API 调试功能,再映射到 Python 爬虫技术,重点分享三大技巧:利用 Postman 生成请求骨架、通过 Session 管理 Cookie 和 User-Agent,以及集成代理 IP 提升稳定性。以票务信息采集为例,展示完整实现流程,探讨其在抗封锁、团队协作等方面的价值,帮助开发者快速构建生产级爬虫代码。
226 1
把Postman调试脚本秒变Python采集代码的三大技巧
|
6月前
|
人工智能 数据挖掘 vr&ar
虚拟现实:建筑设计的新革命
虚拟现实:建筑设计的新革命
187 22
|
存储
烧录树莓派操作系统镜像的详细操作步骤
本文介绍了在树莓派上烧录操作系统镜像的详细步骤,包括准备工具、下载系统镜像、使用烧录软件等关键环节,帮助用户顺利完成树莓派的初始化配置。
1824 6
|
数据安全/隐私保护 Kotlin
Kotlin - 类成员
Kotlin - 类成员
123 1
|
Java 开发工具 Android开发
Kotlin语法笔记(26) -Kotlin 与 Java 共存(1)
Kotlin语法笔记(26) -Kotlin 与 Java 共存(1)
138 2
|
Java 编译器 Android开发
Kotlin语法笔记(28) -Kotlin 与 Java 混编
Kotlin语法笔记(28) -Kotlin 与 Java 混编
211 2
|
Java 编译器 Android开发
Kotlin语法笔记(28) -Kotlin 与 Java 混编
本系列教程详细讲解了Kotlin语法,适合需要深入了解Kotlin的开发者。对于希望快速学习Kotlin的用户,推荐查看“简洁”系列教程。本文档重点介绍了Kotlin与Java混编的技巧,包括代码转换、类调用、ProGuard问题、Android library开发建议以及在Kotlin和Java之间互相调用的方法。
222 1

热门文章

最新文章

下一篇
oss云网关配置