kotlin中聲明屬性的時(shí)候是要要求初始化的,否則就會(huì)編輯器就會(huì)報(bào)錯(cuò)屡穗,但是在開(kāi)發(fā)中我們并不是立即初始化屬性,比如定義控件名稱的時(shí)候,我們不會(huì)在定義時(shí)就fingViewById來(lái)初始化控件咆瘟,而是在頁(yè)面生命周期函數(shù)里面來(lái)初始化控件。kotlin為開(kāi)發(fā)者們提供了延遲初始化的方案诽里。
接下來(lái)讓我們講講kotlin中兩種延遲初始化的技術(shù)
lateinit
看看例子
class AttriKotlin {
//val song :Int//Property must be initialized or be abstract
val song :Int = 0
var biao :String = "biao"
}
在上面的例子中我們可以到編輯器報(bào)錯(cuò)的內(nèi)容是指該屬性必須初始化或者抽象袒餐。我們沒(méi)必要抽象又想不立即初始化,所以讓我們看看用上lateinit
class AttriKotlin {
//'lateinit' modifier is not allowed on properties of primitive types
lateinit var song:Int
//'lateinit' modifier is not allowed on properties of nullable types
lateinit var song:String?
lateinit var biao:String//這樣子是ok的
}
從上面例子中可以看到lateinit不能修飾基本數(shù)據(jù)類(lèi)型谤狡,也不能修飾可為空的屬性灸眼,原因在于Kotlin會(huì)使用null來(lái)對(duì)每一個(gè)用lateinit修飾的屬性做初始化,而基礎(chǔ)類(lèi)型是沒(méi)有null類(lèi)型墓懂,所以無(wú)法使用lateinit焰宣。
再來(lái)項(xiàng)目中經(jīng)常需要用到的,eg
class lazyClass(): Activity() {
lateinit var textView:TextView
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
textView = findViewById(R.id.getEdit_btn)
}
}
小結(jié):
- lateinit只能用于var捕仔,所以經(jīng)常是lateinit var連用
- lateinit只能用來(lái)修飾類(lèi)屬性匕积,不能用來(lái)修飾局部變量
- lateinit不能修飾基本數(shù)據(jù)類(lèi)型,如果int(kotlin基本類(lèi)型:Double榜跌、Int闪唆、Float、Long钓葫、Short苞氮、Byte)
- lateinit用于只能生命周期流程中進(jìn)行獲取或者初始化的變量,比如 Android 的 onCreate()
by lazy
延遲加載有幾個(gè)好處瓤逼。首先由于加載時(shí)機(jī)推遲到了變量被訪問(wèn)時(shí)笼吟,因此它可以提高應(yīng)用的啟動(dòng)速度。相比于使用 Kotlin 開(kāi)發(fā)服務(wù)端應(yīng)用霸旗,這一特性對(duì)開(kāi)發(fā) Android 應(yīng)用而言特別有用贷帮。對(duì)于 Android 應(yīng)用來(lái)說(shuō),我們想要減少應(yīng)用啟動(dòng)時(shí)間诱告,這樣用戶可以更快看到應(yīng)用的內(nèi)容撵枢,而不是干等著看啟動(dòng)加載頁(yè)面。
其次精居,這樣的延遲加載也有更高的內(nèi)存效率锄禽,因?yàn)槲覀冎辉谒徽{(diào)用時(shí)才將資源加載進(jìn)內(nèi)存。在 Android 這樣的移動(dòng)平臺(tái)上靴姿,內(nèi)存的使用是非常重要的沃但。因?yàn)槭謾C(jī)資源是共享的且有限的。
延遲加載也是封裝初始化邏輯的好方法:
val views: RectF by lazy {
RectF(0f, 0f, width.toFloat(), height.toFloat())
}
只有當(dāng) bounds 變量第一次被引用時(shí)佛吓,將會(huì)使用 view 的當(dāng)前寬和高的值來(lái)創(chuàng)建 RectF宵晚,這樣我們就不需要一開(kāi)始顯式的創(chuàng)建 RectF垂攘,然后把它設(shè)置給 views。
- lazy只能修飾val
- 應(yīng)用于單例模式(if-null-then-init-else-return)淤刃,而且當(dāng)且僅當(dāng)變量被第一次調(diào)用的時(shí)候晒他,委托方法才會(huì)執(zhí)行。
eg:
//by lazy是單例模式的(if null init else return),而且當(dāng)且僅當(dāng)變量第一次被初始化時(shí)逸贾,委托方法(by lazy)才會(huì)被調(diào)用
val name by lazy {
println("只調(diào)用一次")
"biao"
}
val lazyClass = LazyClass()
val name = lazyClass.name
val name1 = lazyClass.name
println("lazy-name:$name")
println("lazy-name:$name1")
result:
只調(diào)用一次
lazy-name:biao
lazy-name:biao
從上面例子可以看出屬性name在第一次被調(diào)用的時(shí)候走了委托方法去初始化陨仅,之后都是拿屬性的getter
eg:
class LazyClass: Activity() {
//kotlin 封裝:
fun <V : View> Activity.bindView(id: Int): Lazy<V> = lazy {
viewFinder(id) as V
}
//acitivity中擴(kuò)展調(diào)用
private val viewFinder: Activity.(Int) -> View?
get() = { findViewById(it) }
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
//在activity中的使用姿勢(shì)
val mTextView by bindView<TextView>(R.id.edit_query)
mTextView.text="執(zhí)行到我時(shí),才會(huì)進(jìn)行控件初始化"
}
}