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