網站首頁 編程語言 正文
原理
拓展函數是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方法。拓展函數的this指針實際上是這個生成方法的第一個參數:
/* compiled from: MainActivity.kt */ public final class MainActivityKt { ?? ?public static final void toast(Context $this$toast, String msg) { ?? ??? ?//參數判空 ?? ? ? ?... ?? ? ? ?// 拓展函數代碼 ?? ? ? ?Toast.makeText($this$toast, msg, 0).show(); ?? ?} }
所以這個this指針實際上是由函數調用的地方傳入的對象引用:
private final void foo(Context context) { ? ? MainActivityKt.toast(context, "hello world"); }
限制
知道了拓展函數的實現原理之后我們就能從原理去理解拓展函數的種種限制.
不能訪問私有成員
由于編譯成java之后,生成的拓展方法實際是靠第一個參數出入對象引用,然后使用這個對象引用去調用對象的方法。因此我們并沒有權限在拓展函數里面調用私有方法:
class TestClass { ? ? fun publicFun() {} ? ? private fun privateFun() {} } fun TestClass.extFun() { ? ? publicFun() // 正確,可以調用公有方法 ? ? privateFun() // 錯誤,不能調用私有方法 }
拓展函數不能實現多態
由于拓展函數并不是真的給類增加一個成員函數,所以父類和子類的同名拓展函數并沒有多態的特性。
例如我們為父類和子類拓展同一個foo()函數:
open class Parent class Child : Parent() fun Parent.foo() { ? ? println("parent") } fun Child.foo() { ? ? println("child") }
然后只要將子類轉換成父類,調用的拓展函數就是父類的拓展函數:
val child = Child() child.foo() (child as Parent).foo() // 輸出: // child // parent
成員函數優先級高,拓展函數不能實現重寫
當拓展函數與類本身或者父類的成員函數相同,在實際調用的時候會優先調用成員函數,并不會出現類似重寫的效果.
例如我們為一個類編寫了一個與成員函數相同的拓展函數,實際優先調用類成員函數:
open class Parent { ? ? fun foo() { ? ? ? ? println("foo") ? ? } } fun Parent.foo() { ? ? println("parent") } Parent().foo() // 輸出: // foo
就算是為子類編寫了一個與父類成員函數相同的拓展函數,也會優先調用父類的成員函數:
open class Parent { ? ? fun foo() { ? ? ? ? println("foo") ? ? } } class Child : Parent() fun Child.foo() { ? ? println("child") } Child().foo() // 輸出: // foo 關閉
為什么要使用Kotlin中的擴展函數
我們都知道在Koltin這門語言可以與Java有非常好的互操作性,所以擴展函數這個新特性可以很平滑與現有Java代碼集成。甚至純Kotlin的項目都可以基于Java庫,甚至Android中的一些框架庫,第三方庫來構建。擴展函數非常適合Kotlin和Java語言混合開發模式。在很多公司一些比較穩定良好的庫都是Java寫,也完全沒必要去用Kotlin語言重寫。但是想要擴展庫的接口和功能,這時候擴展函數可能就會派上用場。使用Kotlin的擴展函數還有一個好處就是沒有副作用,不會對原有庫代碼或功能產生影響。先來看下擴展函數長啥樣
給TextView設置加粗簡單的例子
//擴展函數定義 fun TextView.isBold() = this.apply { paint.isFakeBoldText = true } ???????//擴展函數調用 activity.find<TextView>(R.id.course_comment_tv_score).isBold()
總結
原文鏈接:https://blog.islinjw.cn/2022/01/25/Kotlin原理-拓展函數/
相關推薦
- 2022-09-30 詳解OpenCV執行連通分量標記的方法和分析_python
- 2023-01-03 在C語言中getchar的使用方法和讀取規則講解_C 語言
- 2022-04-20 python數據類型中的字符串你了解多少_python
- 2022-12-14 Oracle?REGEXP_LIKE模糊查詢用法例子_oracle
- 2022-07-09 C++深入探究用NULL來初始化空指針是否合適_C 語言
- 2022-11-15 python運行cmd命令行的3種方法總結_python
- 2022-09-30 Python實現圖像增強_python
- 2023-07-14 react實現拖拽功能
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支