前言
Kotlin
是一种在Java
虚拟机上运行的静态类型编程语言,被称之为Android
世界的Swift
,在Google
I/O2017中,Google
宣布Kotlin
成为Android
官方开发语言
代理属性
Kotlin中的代理属性又被称为属性委托,属性的操作就是set/get
函数的调用,委托属性会用自己的setValue/getValue
去大理源属性的实现,委托类需要根据属性的可变类型,实现对应的setValue/getValue
方法
属性委托的语法:
val/var <属性名>:属性类型 by <表达式>
标准委托
Kotlin所提供的一些用于代理属性的工厂方法
延迟属性
使用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知识整理