Kotlin原理詳析之拓展函數(shù)
原理
拓展函數(shù)是kotlin里一個(gè)比較常用的特性,例如我們可以給Context拓展一個(gè)toast方法:
// MainActivity.kt
fun Context.toast(msg: String) {
? ? Toast.makeText(this, msg, Toast.LENGTH_SHORT).show()
}
private fun foo(context: Context) {
? ? context.toast("hello world")
}它的原理其實(shí)很簡單,就是生成了一個(gè)toast方法。拓展函數(shù)的this指針實(shí)際上是這個(gè)生成方法的第一個(gè)參數(shù):
/* compiled from: MainActivity.kt */
public final class MainActivityKt {
?? ?public static final void toast(Context $this$toast, String msg) {
?? ??? ?//參數(shù)判空
?? ? ? ?...
?? ? ? ?// 拓展函數(shù)代碼
?? ? ? ?Toast.makeText($this$toast, msg, 0).show();
?? ?}
}所以這個(gè)this指針實(shí)際上是由函數(shù)調(diào)用的地方傳入的對象引用:
private final void foo(Context context) {
? ? MainActivityKt.toast(context, "hello world");
}限制
知道了拓展函數(shù)的實(shí)現(xiàn)原理之后我們就能從原理去理解拓展函數(shù)的種種限制.
不能訪問私有成員
由于編譯成java之后,生成的拓展方法實(shí)際是靠第一個(gè)參數(shù)出入對象引用,然后使用這個(gè)對象引用去調(diào)用對象的方法。因此我們并沒有權(quán)限在拓展函數(shù)里面調(diào)用私有方法:
class TestClass {
? ? fun publicFun() {}
? ? private fun privateFun() {}
}
fun TestClass.extFun() {
? ? publicFun() // 正確,可以調(diào)用公有方法
? ? privateFun() // 錯(cuò)誤,不能調(diào)用私有方法
}拓展函數(shù)不能實(shí)現(xiàn)多態(tài)
由于拓展函數(shù)并不是真的給類增加一個(gè)成員函數(shù),所以父類和子類的同名拓展函數(shù)并沒有多態(tài)的特性。
例如我們?yōu)楦割惡妥宇愅卣雇粋€(gè)foo()函數(shù):
open class Parent
class Child : Parent()
fun Parent.foo() {
? ? println("parent")
}
fun Child.foo() {
? ? println("child")
}然后只要將子類轉(zhuǎn)換成父類,調(diào)用的拓展函數(shù)就是父類的拓展函數(shù):
val child = Child() child.foo() (child as Parent).foo() // 輸出: // child // parent
成員函數(shù)優(yōu)先級高,拓展函數(shù)不能實(shí)現(xiàn)重寫
當(dāng)拓展函數(shù)與類本身或者父類的成員函數(shù)相同,在實(shí)際調(diào)用的時(shí)候會(huì)優(yōu)先調(diào)用成員函數(shù),并不會(huì)出現(xiàn)類似重寫的效果.
例如我們?yōu)橐粋€(gè)類編寫了一個(gè)與成員函數(shù)相同的拓展函數(shù),實(shí)際優(yōu)先調(diào)用類成員函數(shù):
open class Parent {
? ? fun foo() {
? ? ? ? println("foo")
? ? }
}
fun Parent.foo() {
? ? println("parent")
}
Parent().foo()
// 輸出:
// foo就算是為子類編寫了一個(gè)與父類成員函數(shù)相同的拓展函數(shù),也會(huì)優(yōu)先調(diào)用父類的成員函數(shù):
open class Parent {
? ? fun foo() {
? ? ? ? println("foo")
? ? }
}
class Child : Parent()
fun Child.foo() {
? ? println("child")
}
Child().foo()
// 輸出:
// foo
關(guān)閉為什么要使用Kotlin中的擴(kuò)展函數(shù)
我們都知道在Koltin這門語言可以與Java有非常好的互操作性,所以擴(kuò)展函數(shù)這個(gè)新特性可以很平滑與現(xiàn)有Java代碼集成。甚至純Kotlin的項(xiàng)目都可以基于Java庫,甚至Android中的一些框架庫,第三方庫來構(gòu)建。擴(kuò)展函數(shù)非常適合Kotlin和Java語言混合開發(fā)模式。在很多公司一些比較穩(wěn)定良好的庫都是Java寫,也完全沒必要去用Kotlin語言重寫。但是想要擴(kuò)展庫的接口和功能,這時(shí)候擴(kuò)展函數(shù)可能就會(huì)派上用場。使用Kotlin的擴(kuò)展函數(shù)還有一個(gè)好處就是沒有副作用,不會(huì)對原有庫代碼或功能產(chǎn)生影響。先來看下擴(kuò)展函數(shù)長啥樣
給TextView設(shè)置加粗簡單的例子
//擴(kuò)展函數(shù)定義
fun TextView.isBold() = this.apply {
paint.isFakeBoldText = true
}
???????//擴(kuò)展函數(shù)調(diào)用
activity.find<TextView>(R.id.course_comment_tv_score).isBold()
總結(jié)
到此這篇關(guān)于Kotlin原理詳析之拓展函數(shù)的文章就介紹到這了,更多相關(guān)Kotlin原理拓展函數(shù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Android RecyclerView區(qū)分視圖類型的Divider的實(shí)現(xiàn)
本篇文章主要介紹了Android RecyclerView區(qū)分視圖類型的Divider的實(shí)現(xiàn),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-04-04
Android Filterable實(shí)現(xiàn)Recyclerview篩選功能的示例代碼
這篇文章主要介紹了Android Filterable實(shí)現(xiàn)Recyclerview篩選功能的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-02-02
Android開發(fā)筆記XML數(shù)據(jù)解析方法及優(yōu)缺點(diǎn)
XML數(shù)據(jù)是一種常見的數(shù)據(jù)格式,Android開發(fā)中需要對其進(jìn)行解析。常用的XML解析方式有DOM、SAX、Pull和Json等,每種方式都有其優(yōu)缺點(diǎn)。開發(fā)者可以根據(jù)具體需求選擇合適的解析方式,提高數(shù)據(jù)解析效率和性能2023-05-05
Android實(shí)現(xiàn)拍照添加時(shí)間水印
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)拍照添加時(shí)間水印,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03
android實(shí)現(xiàn)自動(dòng)發(fā)送郵件
這篇文章主要為大家詳細(xì)介紹了android實(shí)現(xiàn)自動(dòng)發(fā)送郵件,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-07-07
Android編程實(shí)現(xiàn)webview執(zhí)行l(wèi)oadUrl時(shí)隱藏鍵盤的workround效果
這篇文章主要介紹了Android編程實(shí)現(xiàn)webview執(zhí)行l(wèi)oadUrl時(shí)隱藏鍵盤的workround效果,較為詳細(xì)的分析了執(zhí)行l(wèi)oadUrl時(shí)隱藏鍵盤的workround具體步驟與兩種實(shí)現(xiàn)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-10-10

