简述
在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关键字修饰
接下来看下Kotlin 标准库提供的委托
例1:lazy()是接受一个 lambda 并返回一个Lazy 实例的函数,返回的实例可以作为实现延迟属性的委托: 第一次调用get()会执行已传递给lazy()的 lamda 表达式并记录结果, 后续调用get()只是返回记录的结果。
lazy测试
var类型属性不能设置为延迟加载属性,因为在lazy中并没有setValue(…)方法
lazy是线程安全的。如果在不考虑多线程问题或者想提高更多的性能,也可以使
用 lazy(LazyThreadSafeMode.NONE){ … }
例2:Observable 使用是通过类Delegates.observable()调用的,需要两个参数:第一个是初始化值, 第二个是属性值变化事件的响应器(handler)
var name: String by Delegates.observable("demo",{ kProperty , oldName , newName-> println("kProperty:${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异常,说该值必须初始化。
notNull源码
例5:map映射,Kotlin给咱们提供了 map到 对象属性的映射,主要用于json解析中,这种委托的写法和上面的不太一样,首先看下代码
class User(val map: Map) { val name: String by map val age: Int by map }
User类接受一个Map集合,会自动将 Map中key 为name的值赋给name, age为key的值赋给age。这样用起来确实挺简单方便。
总结
委托属性在实际开发中还是会应用很多的,这属于kotlin中的重点之一吧,需要好好研究下,到此为止,Kotlin中的类与对象就告一段落,不知道你们学的怎么样,下期继续学习kotlin中的又一个特点:空安全。