java中一些属性的具有相同的行为怎么办,抽象出类然后再去依赖调用,而在Kotlin中只需要一个by关键字就能省去繁琐的依赖。属性的委托主要是统一实现了对属性的set,get。Kotlin 标准库还为几种有用的委托提供了工厂方法:
延迟属性(lazy properties): 其值只在首次访问时计算,
可观察属性(observable properties): 监听器会收到有关此属性变更的通知,
把多个属性储存在一个映射(map)中,而不是每个存在单独的字段中。
下面将通过编译器的代码例子详细学习。
例1:声明方式
val/var <属性名>: <类型> by <表达式>
例2:by <表达式> 这个表达式怎么写,下面一个例子
class ByDemo{
operator fun getValue(thisRef: Any?,property: KProperty<*>) : <类型>{
return <类型>
}
operator fun setValue(thisRef: Any?,property: KProperty<*>,value:<类型>){
print()
}
}
解释下这个例子:
定义一个ByDemo类,然后再定义两个方法getValue(),setValue() , 方法需要operator关键字修饰,thisRef: Any? :thisRef是当前属性的对象,property: KProperty<*>:property是当前属性,property.name 属性名称,value:<类型> 这个就是当前属性的值和类型。
在定义委托表达式需要注意的点:
对于只读属性(也就是说val属性), 它的委托必须提供一个名为getValue()的函数。
thisRef: Any? 必须是该属性当前类或者基类(Any是任何类的基类)。
property: KProperty<>这个参数的类型必须是 KProperty<> , 或者是它的基类。
getValue()返回值类型必须与属性类型相同(或者是它的子类型)。
value:<类型>这个参数的类型必须与属性类型相同, 或者是它的基类
方法名不能改必须是getValue、setValue,并且必须用operator关键字修饰
var类型属性不能设置为延迟加载属性,因为在lazy中并没有setValue(…)方法
lazy是线程安全的。如果在不考虑多线程问题或者想提高更多的性能,也可以使
用 lazy(LazyThreadSafeMode.NONE){ … }
例2:Observable 使用是通过类Delegates.observable()调用的,需要两个参数:第一个是初始化值, 第二个是属性值变化事件的响应器(handler)
var name: String by Delegates.observable(“demo”,{
kProperty , oldName , newName->
println(“kProperty:
k
P
r
o
p
e
r
t
y
.
n
a
m
e
∣
o
l
d
N
a
m
e
:
{kProperty.name}| oldName:
kProperty.name∣oldName:oldName| newName:$newName”)
})
“demo” 为第一个参数,用意是给name赋值。
"demo"后面大括号内容为第二个参数,表达式有三个值:每次给name赋值都会回调它,就跟观察者一样,kProperty是当前属性,oldName是它的旧值,newName是它的新值。这里都是变量名可以随意起名字,你也可以这样 { a, b, c -> println( a+b+c) } 当然这样命名不规范不推荐,这就是为了说明可以改。
例3:Vetoable 使用也是通过 类Delegates.vetoable()调用,同样也是两个参数,第一个是初始化值, 第二个是属性值变化事件的响应器(handler),是可观察属性(Observable)的一个特例,不同的是给属性赋值的时候会加以判断,是否要将新值赋于该变量。
var name: String by Delegates.vetoable("demo",{
kProperty , oldName , newName->
println("kProperty:${kProperty.name}| oldName:$oldName| newName:$newName")
newName.contains("demo") //判断新的值是否包含demo字符,有才会赋值
})
例4:notNull 使用 通过类 Delegates.notNull() 调用,在java中我们会遇到很多的null判断,空的话会抛出异常,Kotlin为了简化该操作,追求高效率的工作,帮我们实现了该方法,下面看下源码就明白了,在value等于null的时候抛出IllegalStateException异常,说该值必须初始化。