前言
Kotlin 中類的擴(kuò)展方法并不是在原類的內(nèi)部進(jìn)行拓展,通過反編譯為Java代碼,可以發(fā)現(xiàn),其原理是使用裝飾模式,對源類實(shí)例的操作和包裝,其實(shí)際相當(dāng)于我們在 Java中定義的工具類方法,并且該工具類方法是使用調(diào)用者為第一個參數(shù)的,然后在工具方法中操作該調(diào)用者;
理論上來說,擴(kuò)展函數(shù)很簡單,它就是一個類的成員函數(shù),不過定義在類的外面。讓我們來添加一個方法,來計算一個字符串的最后一個字符:
package strings/** * @author :Reginer in 2018/5/22 21:04. * 聯(lián)系方式:QQ:282921012 * 功能描述: */fun String.lastChar(): Char = get(this.length - 1)
要做的,就是把要擴(kuò)展的類或者接口的名稱,放到即將添加的函數(shù)前面。這個類的名稱被稱為接收者類型;用來調(diào)用這個擴(kuò)展函數(shù)的對象,叫做接收者對象。
接收者類型是由擴(kuò)展函數(shù)定義的,接收對象是該類型的一個實(shí)例
可以像調(diào)用類的成員函數(shù)一樣去調(diào)用這個函數(shù):
println("Kotlin".lastChar())>n從某種意義上說,現(xiàn)在已經(jīng)為String類添加了自己的方法。不管String類是用Java、Kotlin,或者像Groovy的其他JVM語言編寫的,只要它會編譯為Java類,就可以為這個類添加自己的擴(kuò)展。
在這個擴(kuò)展函數(shù)中,可以像其他成員函數(shù)一樣用this。也可以像普通的成員函數(shù)一樣,省略它:
fun String.lastChar(): Char = get(this.length - 1)
注意,擴(kuò)展函數(shù)并不允許打破它的封裝性。和在類內(nèi)部定義的方法不同的是,擴(kuò)展函數(shù)不能訪問私有的或者是受保護(hù)的成員。
3.3.1導(dǎo)入和擴(kuò)展函數(shù)
對于定義的一個擴(kuò)展函數(shù),它不會自動地在整個項(xiàng)目范圍內(nèi)生效。相反,如果要使用它,需要進(jìn)行導(dǎo)入,就像其他任何的類或者函數(shù)一樣。這是為了避免偶然性的命名沖突。Kotlin允許用和導(dǎo)入類一樣的語法來導(dǎo)入單個的函數(shù):
import strings.lastChar//星號導(dǎo)入import strings.*
3.3.2在Java中調(diào)用擴(kuò)展函數(shù)
其實(shí),擴(kuò)展函數(shù)是靜態(tài)函數(shù),它把調(diào)用對象作為了它的第一個參數(shù)。調(diào)用擴(kuò)展函數(shù),不會創(chuàng)建適配的對象或者任何運(yùn)行時的額外消耗。
這使得從Java中調(diào)用Kotlin的擴(kuò)展函數(shù)變得非常簡單:調(diào)用這個靜態(tài)函數(shù),然后把接收對象作為第一個參數(shù)傳進(jìn)去即可。假設(shè)它聲明在一個叫做StringUtil.kt的文件中:
char c = StringUtil.lastChar("Java");和Kotlin版本比較起來,可讀性略差。
3.3.3作為擴(kuò)展函數(shù)的工具函數(shù)
現(xiàn)在,可以寫一個joinToString函數(shù)的終極版本了:
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()}因?yàn)閿U(kuò)展函數(shù)無非就是靜態(tài)函數(shù)的一個高效語法糖,可以使用更具體的類型來作為接收者類型,而不是一個類。假設(shè)想要一個join函數(shù),只能由字符串的集合來觸發(fā):
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()}如果是用其他類型的對象列表來調(diào)用會報錯。
3.3.4不可重寫的擴(kuò)展函數(shù)
擴(kuò)展函數(shù)并不是類的一部分,它是聲明在類之外的。擴(kuò)展函數(shù)并不存在重寫,因?yàn)镵otlin會把它們當(dāng)做靜態(tài)函數(shù)對待。
3.3.5擴(kuò)展屬性
val String.lastChar: Char get() = get(this.length - 1)
和擴(kuò)展函數(shù)一樣,擴(kuò)展屬性也像接收者的一個普通成員屬性一樣。
這里必須定義getter函數(shù),因?yàn)闆]有支持字段,因此沒有默認(rèn)的getter的實(shí)現(xiàn)。同理,初始化也不可以,因?yàn)闆]有地方存儲初始值。
如果在StringBuilder上定義一個相同的屬性,可以置為var,因?yàn)镾tringBuilder的內(nèi)容是可變的:
var StringBuilder.lastChar: Char get() = get(length - 1) set(value) { this.setCharAt(length - 1, value) }可以像訪問成員屬性一樣訪問它:
println("Kotlin".lastChar)> nval sb = StringBuilder("Kotlin?")sb.lastChar = '!'println(sb)> Kotlin!注意,當(dāng)需要從Java中訪問擴(kuò)展屬性的時候,應(yīng)該顯式地調(diào)用它的getter函數(shù):StringUtil.getLastChar("Java");
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,如果有疑問大家可以留言交流,謝謝大家對VeVb武林網(wǎng)的支持。
新聞熱點(diǎn)
疑難解答
圖片精選