Koltin中的代理属性 - 标准委托

2021-06-15  本文已影响0人  盛世光阴

前言

Kotlin是一种在Java虚拟机上运行的静态类型编程语言,被称之为Android世界的Swift,在GoogleI/O2017中,Google宣布Kotlin成为Android官方开发语言

代理属性

Kotlin中的代理属性又被称为属性委托,属性的操作就是set/get函数的调用,委托属性会用自己的setValue/getValue去大理源属性的实现,委托类需要根据属性的可变类型,实现对应的setValue/getValue方法

属性委托的语法:
val/var <属性名>:属性类型 by <表达式>

标准委托

Kotlin所提供的一些用于代理属性的工厂方法

延迟属性
使用lazy实现延迟属性的委托,只在第一次首次访问的时候会计算

Kotlin中的lateinit、lazy关键字

可观察属性
每当我们给属性赋值的时候,就会在赋值后调用表达式中代码块,用于监听属性内容的变更

var value:String by Delegates.observable("initValue"){
        pro,old,new ->
    Log.e("mike","----$pro----$old----$new")
}
value = "aaa"
value = "aaa"
value = "bbb"
//打印结果
----var com.mike.loginmvvm.login.ui.LoginActivity.value: kotlin.String----initValue----aaa
----var com.mike.loginmvvm.login.ui.LoginActivity.value: kotlin.String----aaa----aaa
----var com.mike.loginmvvm.login.ui.LoginActivity.value: kotlin.String----aaa----bbb

非空属性代理

var data:String by Delegates.notNull()
Log.e("mike","----$data----")
//打印结果
Caused by: java.lang.IllegalStateException: Property data should be initialized before get.

查看它的源码发现,如果数据为空的时候会抛出异常

public override fun getValue(thisRef: Any?, property: KProperty<*>): T {
    return value ?: throw IllegalStateException("Property ${property.name} should be initialized before get.")
}

那么它和lateinit很相似,都是可以绕过编译器检查,都是初始化前直接使用会有异常抛出,就使用而言notNull可以用于基本数据类型,但lateinit只能用于引用数据类型

lateinit会生成这样的代码,这样也能看出之所以lateinit不支持基本数据类型,也是因为基本数据类型在java中存在默认值的,所以不存在稍后初始化这种情况

lateinit var data:String
//java代码
public String data;
public void get(){
    if (data == null){
        Intrinsics.throwUninitializedPropertyAccessException("data");
    }
    return data;
}

看下相同情况下的Delegates.notNull(),会产生一个包装类的对象将数据进行包装

var value by Delegates.notNull<String>()

//会生成这样的伪代码
public NotNullVar value$delegate = new NotNullVar()
public String getValue(){
    return value$delegate.getValue();
}

所以当我们在选择Delegates.notNull()lateinit的时候,如果是引用类型的需要延迟初始化的情况下推荐使用lateinit,开销相对较小,如果是基本数据类型就使用Delegates.notNull()

属性值拦截
Delegates.observable可以查看数据的情况,在数据变化过后进行回调,vetoable则可以在数据提交之前进行回调,从而决定是否需要此次变更,当返回false时表示丢弃此次变更

var version: Version by Delegates.vetoable(Version(0)) { pro, old, new ->
    return@vetoable new.code > old.code
}

欢迎关注Mike的简书

Android知识整理

上一篇下一篇

猜你喜欢

热点阅读