抬頭
---
屬性相關
* 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
- 0-發現
- AndroidInterview-Q-A
- Android能讓你少走彎路的干貨整理
- LearningNotes
- temp
- temp11
- 部分地址
- 0-待辦任務
- 待補充列表
- 0-未分類
- AndroidView事件分發與滑動沖突處理
- Spannable
- 事件分發機制詳解
- 1-Java
- 1-Java-01基礎
- 未歸檔
- 你應該知道的JDK知識
- 集合框架
- 1-Java-04合集
- Java之旅0
- Java之旅
- JAVA之旅01
- JAVA之旅02
- JAVA之旅03
- JAVA之旅04
- JAVA之旅05
- JAVA之旅06
- JAVA之旅07
- JAVA之旅08
- JAVA之旅09
- java之旅1
- JAVA之旅10
- JAVA之旅11
- JAVA之旅12
- JAVA之旅13
- JAVA之旅14
- JAVA之旅15
- JAVA之旅16
- JAVA之旅17
- JAVA之旅18
- JAVA之旅19
- java之旅2
- JAVA之旅20
- JAVA之旅21
- JAVA之旅22
- JAVA之旅23
- JAVA之旅24
- JAVA之旅25
- JAVA之旅26
- JAVA之旅27
- JAVA之旅28
- JAVA之旅29
- java之旅3
- JAVA之旅30
- JAVA之旅31
- JAVA之旅32
- JAVA之旅33
- JAVA之旅34
- JAVA之旅35
- 1-Java-05辨析
- HashMapArrayMap
- Java8新特性
- Java8接口默認方法
- 圖解HashMap(1)
- 圖解HashMap(2)
- 2-Android
- 2-Android-1-基礎
- View繪制流程
- 事件分發
- AndroidView的事件分發機制和滑動沖突解決
- 自定義View基礎
- 1-安卓自定義View基礎-坐標系
- 2-安卓自定義View基礎-角度弧度
- 3-安卓自定義View基礎-顏色
- 自定義View進階
- 1-安卓自定義View進階-分類和流程
- 10-安卓自定義View進階-Matrix詳解
- 11-安卓自定義View進階-MatrixCamera
- 12-安卓自定義View進階-事件分發機制原理
- 13-安卓自定義View進階-事件分發機制詳解
- 14-安卓自定義View進階-MotionEvent詳解
- 15-安卓自定義View進階-特殊形狀控件事件處理方案
- 16-安卓自定義View進階-多點觸控詳解
- 17-安卓自定義View進階-手勢檢測GestureDetector
- 2-安卓自定義View進階-繪制基本圖形
- 3-安卓自定義View進階-畫布操作
- 4-安卓自定義View進階-圖片文字
- 5-安卓自定義View進階-Path基本操作
- 6-安卓自定義View進階-貝塞爾曲線
- 7-安卓自定義View進階-Path完結篇偽
- 8-安卓自定義View進階-Path玩出花樣PathMeasure
- 9-安卓自定義View進階-Matrix原理
- 通用類介紹
- Application
- 2-Android-2-使用
- 2-Android-02控件
- ViewGroup
- ConstraintLayout
- CoordinatorLayout
- 2-Android-03三方使用
- Dagger2
- Dagger2圖文完全教程
- Dagger2最清晰的使用教程
- Dagger2讓你愛不釋手-終結篇
- Dagger2讓你愛不釋手-重點概念講解、融合篇
- dagger2讓你愛不釋手:基礎依賴注入框架篇
- 閱讀筆記
- Glide
- Google推薦的圖片加載庫Glide:最新版使用指南(含新特性)
- rxjava
- 這可能是最好的RxJava2.x入門教程完結版
- 這可能是最好的RxJava2.x入門教程(一)
- 這可能是最好的RxJava2.x入門教程(三)
- 這可能是最好的RxJava2.x入門教程(二)
- 這可能是最好的RxJava2.x入門教程(五)
- 這可能是最好的RxJava2.x入門教程(四)
- 2-Android-3-優化
- 優化概況
- 各種優化
- Android端秒開優化
- apk大小優化
- 內存分析
- 混淆
- 2-Android-4-工具
- adb命令
- 一鍵分析Android的BugReport
- 版本控制
- git
- git章節簡述
- 2-Android-5-源碼
- HandlerThread 源碼分析
- IntentService的使用和源碼分析
- 2-Android-9-辨析
- LRU算法
- 什么是Bitmap
- 常見圖片壓縮方式
- 3-Kotlin
- Kotlin使用筆記1-草稿
- Kotlin使用筆記2
- kotlin特性草稿
- Kotlin草稿-Delegation
- Kotlin草稿-Field
- Kotlin草稿-object
- 4-JavaScript
- 5-Python
- 6-Other
- Git
- Gradle
- Android中ProGuard配置和總結
- gradle使用筆記
- Nexus私服搭建
- 編譯提速最佳實踐
- 7-設計模式與架構
- 組件化
- 組件化探索(OKR)
- 1-參考列表
- 2-1-組件化概述
- 2-2-gradle配置
- 2-3-代碼編寫
- 2-4-常見問題
- 2-9-值得一讀
- 8-數據結構與算法
- 0臨時文件
- 漢諾塔
- 8-數據-1數據結構
- HashMap
- HashMap、Hashtable、HashSet 和 ConcurrentHashMap 的比較
- 遲到一年HashMap解讀
- 8-數據-2算法
- 1個就夠了
- Java常用排序算法(必須掌握的8大排序算法)
- 常用排序算法總結(性能+代碼)
- 必須知道的八大種排序算法(java實現)
- 9-職業
- 閱讀
- 書單
- 面試
- 面試-01-java
- Java面試題全集駱昊(上)
- Java面試題全集駱昊(下)
- Java面試題全集駱昊(中)
- 面試-02-android
- 40道Android面試題
- 面試-03-開源源碼
- Android圖片加載框架最全解析(二),從源碼的角度理解Glide的執行流程
- 面試-07-設計模式
- 面試-08-算法
- 面試-09-其他
- SUMMARY
- 版權說明
- temp111