http://www.reibang.com/p/a70ba6436e75/
http://www.reibang.com/p/306bdc2bac3f
代理模式:為其他對象提供一種代理以控制其他對象對這個(gè)對象的訪問贫堰。
委托屬性的聲明
定義一個(gè)委托屬性的語法是 val/var <property name>: <Type> by <expression>,其中 by 后面的就是屬性的委托远荠。屬性委托不用繼承什么特別的接口亚铁,只要擁有用 operator 修飾的 getValue() 和 setValue() (適用 var)的函數(shù)就可以了。
by lazy的必須用val修飾
類委托
一個(gè)類中定義的方法實(shí)際是調(diào)用另一個(gè)類的對象的方法來實(shí)現(xiàn)的树枫。
//創(chuàng)建接口
interface Base {
fun print()
}
//實(shí)現(xiàn)此接口的被委托的類
class BaseImpl(val x: Int) : Base {
override fun print() { print(x) }
}
//通過關(guān)鍵字 by 建立委托類
class Derived(b: Base) : Base by b
//測試
fun main(args: Array<String>) {
val b = BaseImpl(10)
Derived(b).print() // 輸出 10
}
在 Derived 聲明中直焙,by 子句表示,將 b 保存在 Derived 的對象實(shí)例內(nèi)部砂轻,而且編譯器將會(huì)生成繼承自 Base 接口的所有方法, 并將調(diào)用轉(zhuǎn)發(fā)給 b奔誓。
屬性委托
屬性委托是指一個(gè)類的某個(gè)屬性值不是在類中直接進(jìn)行定義,而是將其托付給一個(gè)代理類搔涝,從而實(shí)現(xiàn)對該類的屬性統(tǒng)一管理厨喂。
val/var 變量名:類型 by 表達(dá)式
by字之后的表達(dá)式就是委托和措,屬性的get set方法將被委托給這個(gè)對象的setVaule getValue方法,屬性委托不必實(shí)現(xiàn)任何接口蜕煌,但必須提供getValue setValue函數(shù)派阱。
import kotlin.reflect.KProperty
// 定義包含屬性委托的類
class Example {
var p: String by Delegate()
}
// 委托的類
class Delegate {
//setValue getValue方法的修飾符 方法名 參數(shù)類型 只能這么寫
operator fun getValue(thisRef: Any?, property: KProperty<*>): String {
return "$thisRef, 這里委托了 ${property.name} 屬性"
}
operator fun setValue(thisRef: Any?, property: KProperty<*>, value: String) {
println("$thisRef 的 ${property.name} 屬性賦值為 $value")
}
}
fun main(args: Array<String>) {
val e = Example()
println(e.p) // 訪問該屬性,調(diào)用 getValue() 函數(shù)
e.p = "Runoob" // 調(diào)用 setValue() 函數(shù)
println(e.p)
}
輸出結(jié)果為:
Example@433c675d, 這里委托了 p 屬性
Example@433c675d 的 p 屬性賦值為 Runoob
Example@433c675d, 這里委托了 p 屬性
標(biāo)準(zhǔn)委托
Kotlin的標(biāo)準(zhǔn)庫中已經(jīng)內(nèi)置了很多工廠方法來實(shí)現(xiàn)屬性委托斜纪。
延遲屬性lazy
lazy()是一個(gè)函數(shù)颁褂,接受一個(gè)lambda表達(dá)式作為參數(shù),返回一個(gè)lazy<T>實(shí)例的函數(shù)傀广,返回的實(shí)例可以作為實(shí)現(xiàn)延遲屬性的委托:第一次調(diào)用get方法會(huì)執(zhí)行已傳遞給lazy的lambda表達(dá)式并記錄結(jié)果颁独,后續(xù)調(diào)用get方法只是返回記錄的結(jié)果。
val lazyValue: String by lazy {
println("computed!") // 第一次調(diào)用輸出伪冰,第二次調(diào)用不執(zhí)行
"Hello"
}
fun main(args: Array<String>) {
println(lazyValue) // 第一次執(zhí)行誓酒,執(zhí)行兩次輸出表達(dá)式
println("--------")
println(lazyValue) // 第二次執(zhí)行,只輸出返回值
}
執(zhí)行輸出結(jié)果:
computed!
Hello
--------
Hello
get方法是線程不安全的贮聂,lazy是線程安全的靠柑,
。吓懈。歼冰。后面還一個(gè)例子看文章。