您好,登錄后才能下訂單哦!
前言
Kotlin 中類的擴展方法并不是在原類的內部進行拓展,通過反編譯為Java代碼,可以發現,其原理是使用裝飾模式,對源類實例的操作和包裝,其實際相當于我們在 Java中定義的工具類方法,并且該工具類方法是使用調用者為第一個參數的,然后在工具方法中操作該調用者;
理論上來說,擴展函數很簡單,它就是一個類的成員函數,不過定義在類的外面。讓我們來添加一個方法,來計算一個字符串的最后一個字符:
package strings /** * @author :Reginer in 2018/5/22 21:04. * 聯系方式:QQ:282921012 * 功能描述: */ fun String.lastChar(): Char = get(this.length - 1)
要做的,就是把要擴展的類或者接口的名稱,放到即將添加的函數前面。這個類的名稱被稱為接收者類型;用來調用這個擴展函數的對象,叫做接收者對象。
接收者類型是由擴展函數定義的,接收對象是該類型的一個實例
可以像調用類的成員函數一樣去調用這個函數:
println("Kotlin".lastChar()) >n
從某種意義上說,現在已經為String類添加了自己的方法。不管String類是用Java、Kotlin,或者像Groovy的其他JVM語言編寫的,只要它會編譯為Java類,就可以為這個類添加自己的擴展。
在這個擴展函數中,可以像其他成員函數一樣用this。也可以像普通的成員函數一樣,省略它:
fun String.lastChar(): Char = get(this.length - 1)
注意,擴展函數并不允許打破它的封裝性。和在類內部定義的方法不同的是,擴展函數不能訪問私有的或者是受保護的成員。
3.3.1導入和擴展函數
對于定義的一個擴展函數,它不會自動地在整個項目范圍內生效。相反,如果要使用它,需要進行導入,就像其他任何的類或者函數一樣。這是為了避免偶然性的命名沖突。Kotlin允許用和導入類一樣的語法來導入單個的函數:
import strings.lastChar //星號導入 import strings.*
3.3.2在Java中調用擴展函數
其實,擴展函數是靜態函數,它把調用對象作為了它的第一個參數。調用擴展函數,不會創建適配的對象或者任何運行時的額外消耗。
這使得從Java中調用Kotlin的擴展函數變得非常簡單:調用這個靜態函數,然后把接收對象作為第一個參數傳進去即可。假設它聲明在一個叫做StringUtil.kt的文件中:
char c = StringUtil.lastChar("Java");
和Kotlin版本比較起來,可讀性略差。
3.3.3作為擴展函數的工具函數
現在,可以寫一個joinToString函數的終極版本了:
fun <T> Collection<T>.joinToString(separator: String = ", ", prefix: String = "", postfix: String = ""): String { val result = StringBuilder(prefix) for ((index, element) in this.withIndex()) { if (index > 0) result.append(separator) result.append(element) } result.append(postfix) return result.toString() }
因為擴展函數無非就是靜態函數的一個高效語法糖,可以使用更具體的類型來作為接收者類型,而不是一個類。假設想要一個join函數,只能由字符串的集合來觸發:
fun Collection<String>.join(separator: String = ", ", prefix: String = "", postfix: String = ""): String { val result = StringBuilder(prefix) for ((index, element) in this.withIndex()) { if (index > 0) result.append(separator) result.append(element) } result.append(postfix) return result.toString() }
如果是用其他類型的對象列表來調用會報錯。
3.3.4不可重寫的擴展函數
擴展函數并不是類的一部分,它是聲明在類之外的。擴展函數并不存在重寫,因為Kotlin會把它們當做靜態函數對待。
3.3.5擴展屬性
val String.lastChar: Char get() = get(this.length - 1)
和擴展函數一樣,擴展屬性也像接收者的一個普通成員屬性一樣。
這里必須定義getter函數,因為沒有支持字段,因此沒有默認的getter的實現。同理,初始化也不可以,因為沒有地方存儲初始值。
如果在StringBuilder上定義一個相同的屬性,可以置為var,因為StringBuilder的內容是可變的:
var StringBuilder.lastChar: Char get() = get(length - 1) set(value) { this.setCharAt(length - 1, value) }
可以像訪問成員屬性一樣訪問它:
println("Kotlin".lastChar) > n val sb = StringBuilder("Kotlin?") sb.lastChar = '!' println(sb) > Kotlin!
注意,當需要從Java中訪問擴展屬性的時候,應該顯式地調用它的getter函數:StringUtil.getLastChar("Java");
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對億速云的支持。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。