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í)很簡(jiǎn)單,就是生成了一個(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)用的地方傳入的對(duì)象引用:
private final void foo(Context context) { ? ? MainActivityKt.toast(context, "hello world"); }
限制
知道了拓展函數(shù)的實(shí)現(xiàn)原理之后我們就能從原理去理解拓展函數(shù)的種種限制.
不能訪問(wèn)私有成員
由于編譯成java之后,生成的拓展方法實(shí)際是靠第一個(gè)參數(shù)出入對(duì)象引用,然后使用這個(gè)對(duì)象引用去調(diào)用對(duì)象的方法。因此我們并沒(méi)有權(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ù)并沒(méi)有多態(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)先級(jí)高,拓展函數(shù)不能實(shí)現(xiàn)重寫(xiě)
當(dāng)拓展函數(shù)與類本身或者父類的成員函數(shù)相同,在實(shí)際調(diào)用的時(shí)候會(huì)優(yōu)先調(diào)用成員函數(shù),并不會(huì)出現(xiàn)類似重寫(xiě)的效果.
例如我們?yōu)橐粋€(gè)類編寫(xiě)了一個(gè)與成員函數(shù)相同的拓展函數(shù),實(shí)際優(yōu)先調(diào)用類成員函數(shù):
open class Parent { ? ? fun foo() { ? ? ? ? println("foo") ? ? } } fun Parent.foo() { ? ? println("parent") } Parent().foo() // 輸出: // foo
就算是為子類編寫(xiě)了一個(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這門語(yǔ)言可以與Java有非常好的互操作性,所以擴(kuò)展函數(shù)這個(gè)新特性可以很平滑與現(xiàn)有Java代碼集成。甚至純Kotlin的項(xiàng)目都可以基于Java庫(kù),甚至Android中的一些框架庫(kù),第三方庫(kù)來(lái)構(gòu)建。擴(kuò)展函數(shù)非常適合Kotlin和Java語(yǔ)言混合開(kāi)發(fā)模式。在很多公司一些比較穩(wěn)定良好的庫(kù)都是Java寫(xiě),也完全沒(méi)必要去用Kotlin語(yǔ)言重寫(xiě)。但是想要擴(kuò)展庫(kù)的接口和功能,這時(shí)候擴(kuò)展函數(shù)可能就會(huì)派上用場(chǎng)。使用Kotlin的擴(kuò)展函數(shù)還有一個(gè)好處就是沒(méi)有副作用,不會(huì)對(duì)原有庫(kù)代碼或功能產(chǎn)生影響。先來(lái)看下擴(kuò)展函數(shù)長(zhǎng)啥樣
給TextView設(shè)置加粗簡(jiǎn)單的例子
//擴(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)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Android RecyclerView區(qū)分視圖類型的Divider的實(shí)現(xiàn)
本篇文章主要介紹了Android RecyclerView區(qū)分視圖類型的Divider的實(shí)現(xiàn),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-04-04Android Filterable實(shí)現(xiàn)Recyclerview篩選功能的示例代碼
這篇文章主要介紹了Android Filterable實(shí)現(xiàn)Recyclerview篩選功能的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-02-02Android樹(shù)形控件的實(shí)現(xiàn)方法
這篇文章主要為大家詳細(xì)介紹了Android樹(shù)形控件的實(shí)現(xiàn)方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-11-11Android Zxing二維碼掃描圖片拉伸問(wèn)題的解決方法
這篇文章主要為大家詳細(xì)介紹了Android Zxing二維碼掃描圖片拉伸問(wèn)題的解決方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-06-06Android開(kāi)發(fā)筆記XML數(shù)據(jù)解析方法及優(yōu)缺點(diǎn)
XML數(shù)據(jù)是一種常見(jiàn)的數(shù)據(jù)格式,Android開(kāi)發(fā)中需要對(duì)其進(jìn)行解析。常用的XML解析方式有DOM、SAX、Pull和Json等,每種方式都有其優(yōu)缺點(diǎn)。開(kāi)發(fā)者可以根據(jù)具體需求選擇合適的解析方式,提高數(shù)據(jù)解析效率和性能2023-05-05Android實(shí)現(xiàn)拍照添加時(shí)間水印
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)拍照添加時(shí)間水印,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03android實(shí)現(xiàn)自動(dòng)發(fā)送郵件
這篇文章主要為大家詳細(xì)介紹了android實(shí)現(xiàn)自動(dòng)發(fā)送郵件,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-07-07Android編程實(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