使用高階函數(shù)時缨睡,每個函數(shù)都是一個對象鸟悴,函數(shù)調(diào)用時還有入棧出棧的開銷。
以lock函數(shù)為例:
fun <T> lock(lock: Lock, body: () -> T): T {
lock.lock()
try {
return body()
} finally {
lock.unlock()
}
}
調(diào)用處
val l = ReentrantLock()
lock(l) {
}
反編譯成JAVA
ReentrantLock l = new ReentrantLock();
KtxKt.lock((Lock)l, (Function0)null.INSTANCE);
lambda編譯成了函數(shù)對象Function0奖年,這樣每次調(diào)用都會生成函數(shù)對象细诸,頻繁創(chuàng)建很不友好。
inline
加入inline關(guān)鍵字
inline fun <T> lock(lock: Lock, body: () -> T): T {
lock.lock()
try {
return body()
} finally {
lock.unlock()
}
}
inline 修飾符影響函數(shù)本?和傳給它的 lambda 表達式:所有這些都將內(nèi)聯(lián)到調(diào)?處陋守。
上述是官方解釋震贵,翻譯成人話:inline修飾的函數(shù)利赋,其本身和參數(shù)lambda表達式的內(nèi)容會直接替換到調(diào)用處。省去了函數(shù)調(diào)用的開銷屏歹,和生成函數(shù)對象隐砸。
調(diào)用處
val l = ReentrantLock()
lock(l) {
}
反編譯成JAVA
ReentrantLock l = new ReentrantLock();
int $i$f$lock = false;
((Lock)l).lock();
try {
boolean var3 = false;
Unit var6 = Unit.INSTANCE;
} finally {
((Lock)l).unlock();
}
加入inline關(guān)鍵字會產(chǎn)生一個新的問題:內(nèi)聯(lián)的 lambda 表達式只能在內(nèi)聯(lián)函數(shù)內(nèi)部調(diào)?或者作為可內(nèi)聯(lián)的參數(shù)傳遞。
inline fun <T> lock(lock: Lock, body: () -> T): T {
lock.lock()
try {
return get(body)
} finally {
lock.unlock()
}
}
fun <T> get(body: () -> T): T {
return body()
}
編譯器會報錯蝙眶,想要將inline的lambda傳遞到非inline函數(shù)季希,需加上noinline關(guān)鍵字禁用內(nèi)聯(lián)。
noinline
inline fun <T> lock(lock: Lock, noinline body: () -> T): T {
lock.lock()
try {
return get(body)
} finally {
lock.unlock()
}
}
fun <T> get(body: () -> T): T {
return body()
}
非局部返回
調(diào)用inline函數(shù)
val l = ReentrantLock()
lock(l) {
return
}
return可使包含該lambda的函數(shù)正常退出
調(diào)用非inline函數(shù)需顯示指定標簽返回
val l = ReentrantLock()
lock(l) {
return@lock
}
crossinline
?些內(nèi)聯(lián)函數(shù)可能調(diào)?傳給它們的不是直接來?函數(shù)體幽纷、?是來?另?個執(zhí)?上下?的 lambda 表達式參數(shù)式塌,例如來?局部對象或嵌套函數(shù)。在這種情況下友浸,該 lambda 表達式中也不允許?局部控制流峰尝。為了標識這種情況,該lambda 表達式參數(shù)需要? crossinline 修飾符標記收恢。
說白了就是不允許直接return返回包含lambda的函數(shù)武学,需顯示指定標簽返回。
inline fun View.setSingleClick(crossinline onclick: (v: View?) -> Unit) {
var lastTime: Long = 0L
this.setOnClickListener {
val currentTime = System.currentTimeMillis()
if (currentTime - lastTime > 500) {
onclick.invoke(it)
}
lastTime = currentTime
}
}
調(diào)用處
btContent.setSingleClick {
return@setSingleClick
}
具體化的類型參數(shù)
內(nèi)聯(lián)函數(shù)?持具體化的類型參數(shù)
inline fun <reified T> ktxClass() = T::class.java
通過reified關(guān)鍵字修飾泛型伦意,可以直接獲取泛型的類型火窒。