<ruby id="bdb3f"></ruby>

    <p id="bdb3f"><cite id="bdb3f"></cite></p>

      <p id="bdb3f"><cite id="bdb3f"><th id="bdb3f"></th></cite></p><p id="bdb3f"></p>
        <p id="bdb3f"><cite id="bdb3f"></cite></p>

          <pre id="bdb3f"></pre>
          <pre id="bdb3f"><del id="bdb3f"><thead id="bdb3f"></thead></del></pre>

          <ruby id="bdb3f"><mark id="bdb3f"></mark></ruby><ruby id="bdb3f"></ruby>
          <pre id="bdb3f"><pre id="bdb3f"><mark id="bdb3f"></mark></pre></pre><output id="bdb3f"></output><p id="bdb3f"></p><p id="bdb3f"></p>

          <pre id="bdb3f"><del id="bdb3f"><progress id="bdb3f"></progress></del></pre>

                <ruby id="bdb3f"></ruby>

                ??一站式輕松地調用各大LLM模型接口,支持GPT4、智譜、豆包、星火、月之暗面及文生圖、文生視頻 廣告
                抬頭 --- 屬性相關 * var、val * 默認實現的get、set * 幕后字段 * 常量放在那里 * lateinit * **委托** --- # 代理模式 在了解Delegation之前,有必要先復習一下代理模式,回顧一下它的使用場景。 不清楚的讀者可以移步[這篇文章](https://link.juejin.im/?target=http%3A%2F%2Fblog.csdn.net%2Fitachi85%2Farticle%2Fdetails%2F50912632)。 這里我要重點引用這篇文章對于應用場景的總結: > 遠程代理:為一個對象在不同的地址空間提供局部代表,這樣系統可以將Server部分的事項隱藏。 > 虛擬代理:使用一個代理對象表示一個十分耗資源的對象并在真正需要時才創建。 > 安全代理:用來控制真實對象訪問時的權限。 > 智能指引:當調用真實的對象時,代理處理另外一些事,比如計算真實對象的引用計數,當該對象沒有引用時,可以自動釋放它;或者訪問一個實際對象時,檢查是否已經能夠鎖定它,以確保其他對象不能改變它。 # [](https://juejin.im/entry/5a1a10a96fb9a0450e75d286?utm_source=gold_browser_extension#Class-Delegation "Class Delegation")Class Delegation 官方文檔給了我們這樣一個例子: > ~~~ > >interface Base { > > fun print() > >} > >class BaseImpl(val x: Int) : Base { > > override fun print() { print(x) } > >} > >class Derived(b: Base) : Base by b > >fun main(args: Array<String>) { > > val b = BaseImpl(10) > > Derived(b).print() // prints 10 > >} > > > > ~~~ 這里通過短句by確定了b這個動態代理,b作為Derived類的對象,編譯器會為它生成所有Base的接口方法。 然后在真正需要代理的時候,把被代理的類的實例作為參數來實例化代理類,然后調用接口方法,則可以實現動態代理。 這個顯然比Java利用反射來實現代理要方便得多。 # [](https://juejin.im/entry/5a1a10a96fb9a0450e75d286?utm_source=gold_browser_extension#Delegated-Properties "Delegated Properties")Delegated Properties 有一些屬性,我們每次需要的時候可以實現他們,但是有種方法可以只需要實現一次。有這樣幾個場景: * lazy properties: 只有第一次訪問時才需要計算的屬性 * observable properties: 監聽變化的屬性 * 把屬性存在一個map里,而不是分散的feild 為了滿足以上需求,Kotlin推出了代理屬性delegate properties: ~~~ class Example { var p: String by Delegate() } ~~~ 與成員p對應的get和set方法都會被Delegate的get和set方法所代理。Delegate不需要實現什么接口,但是必須提供get方法,如果是var類型的,還必須提供set方法。eg: ~~~ class Delegate { operator fun getValue(thisRef: Any?, property: KProperty<*>): String { return "$thisRef, thank you for delegating '${property.name}' to me!" } operator fun setValue(thisRef: Any?, property: KProperty<*>, value: String) { println("$value has been assigned to '${property.name} in $thisRef.'") } } ~~~ 現在屬性p就變成了Delegate的一個實例,讀取p就會調用Delegate的getValue方法,第一個參數表示代理p屬性所在的類的實例,第二個參數則是p屬性本身。例如: ~~~ val e = Example() println(e.p) // will print "Example@33a17727, thank you for delegating ‘p’ to me!" e.p = "NEW" // will print "NEW has been assigned to ‘p’ in Example@33a17727." ~~~ 值得注意的是,getValue和setValue方法前必須添加operater,這是因為Delegate類實際是實現了系統標準庫的接口,所以必須保持一致。 ## [](https://juejin.im/entry/5a1a10a96fb9a0450e75d286?utm_source=gold_browser_extension#標準代理庫 "標準代理庫")標準代理庫 Kotlin標準庫提供了幾個常用的標準代理。老實講,除了lazy目前其他的靈活代理我還沒有發現使用場景。發現了一定回來做補充。 ### [](https://juejin.im/entry/5a1a10a96fb9a0450e75d286?utm_source=gold_browser_extension#lazy "lazy")lazy lazy是一個方法。它可以通過傳入一個lamda函數返回一個Lazy實例用于代理屬性。第一次get調用,會執行傳入lazy()的lamda函數,并且記錄返回值,后續的調用只會返回第一次記錄的值。 例如: ~~~ val lazyValue: String by lazy { println("computed!") "Hello" } fun main(args: Array<String>) { println(lazyValue) println(lazyValue) } ~~~ 打印結果是這樣的: ~~~ computed! Hello Hello ~~~ 如果你想要線程安全,使用 blockingLazy(): 它還是按照同樣的方式工作,但保證了它的值只會在一個線程中計算,并且所有的線程都獲取的同一個值。 用途:在我們生命類的成員的時候,很多時候還不需要初始化,這時,我們就可以用以初始化的構造函數作為lazy的參數,然后形成代理屬性。比如: ~~~ private val bannerAdapter: BannerAdapter by lazy { BannerAdapter() } val viewPager: ViewPager by lazy { ViewPager(context) } private val indicators: LinearLayout by lazy { LinearLayout(context) } private val tvTitle: JumpShowTextView by lazy { JumpShowTextView(context) } private val tvSlogan: JumpShowTextView by lazy { JumpShowTextView(context) } ~~~ ### [](https://juejin.im/entry/5a1a10a96fb9a0450e75d286?utm_source=gold_browser_extension#observable "observable")observable Delegates.observable() 有兩個參數:初始值和用于修改的handler。每次給這個屬性派值(生效)的時候,handler都會被調用。這個Handler有三個參數:被指派的屬性,舊值和新值: ~~~ import kotlin.properties.Delegates class User { var name: String by Delegates.observable("<no name>") { prop, old, new -> println("$old -> $new") } } fun main(args: Array<String>) { val user = User() user.name = "first" user.name = "second" } ~~~ 這個例子打印如下: ~~~ <no name> -> first first -> second ~~~ ### [](https://juejin.im/entry/5a1a10a96fb9a0450e75d286?utm_source=gold_browser_extension#map "map")map Delegates.mapVal() 擁有一個 map 實例并返回一個可以從 map 中讀其中屬性的代理。在應用中有很多這樣的例子,比如解析 JSON 或者做其它的一些 “動態”的事情: ~~~ class User(val map: Map<String, Any?>) { val name: String by Delegates.mapVal(map) val age: Int by Delegates.mapVal(map) } ~~~ 在這個例子中,構造函數持有一個 map : ~~~ val user = User(mapOf ( "name" to "John Doe", "age" to 25 )) ~~~ 代理從這個 map 中取指(通過屬性的名字): ~~~ println(user.name) // Prints "John Doe" println(user.age) // Prints 25 ~~~ var 可以用 mapVar
                  <ruby id="bdb3f"></ruby>

                  <p id="bdb3f"><cite id="bdb3f"></cite></p>

                    <p id="bdb3f"><cite id="bdb3f"><th id="bdb3f"></th></cite></p><p id="bdb3f"></p>
                      <p id="bdb3f"><cite id="bdb3f"></cite></p>

                        <pre id="bdb3f"></pre>
                        <pre id="bdb3f"><del id="bdb3f"><thead id="bdb3f"></thead></del></pre>

                        <ruby id="bdb3f"><mark id="bdb3f"></mark></ruby><ruby id="bdb3f"></ruby>
                        <pre id="bdb3f"><pre id="bdb3f"><mark id="bdb3f"></mark></pre></pre><output id="bdb3f"></output><p id="bdb3f"></p><p id="bdb3f"></p>

                        <pre id="bdb3f"><del id="bdb3f"><progress id="bdb3f"></progress></del></pre>

                              <ruby id="bdb3f"></ruby>

                              哎呀哎呀视频在线观看