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