概述
Kotlin 中类的扩展方法并不是在原类的内部进行拓展,通过反编译为 Java 代码可以发现,其原理是使用装饰模式,对源类实例的操作和包装。其实际相当于在 Java 中定义的工具类方法,并且该工具类方法是使用调用者为第一个参数的,然后在工具方法中操作该调用者。
理论上来说,扩展函数很简单,它就是一个类的成员函数,不过定义在类的外面。让我们来添加一个方法,来计算一个字符串的最后一个字符:
package strings
fun String.lastChar(): Char = get(this.length - 1)
要做的,就是把要扩展的类或者接口的名称,放到即将添加的函数前面。这个类的名称被称为接收者类型;用来调用这个扩展函数的对象,叫做接收者对象。
接收者类型是由扩展函数定义的,接收对象是该类型的一个实例。
可以像调用类的成员函数一样去调用这个函数:
println("Kotlin".lastChar())
从某种意义上说,现在已经为 String 类添加了自己的方法。不管 String 类是用 Java、Kotlin,或者像 Groovy 的其他 JVM 语言编写的,只要它会编译为 Java 类,就可以为这个类添加自己的扩展。
在这个扩展函数中,可以像其他成员函数一样用 this。也可以像普通的成员函数一样,省略它:
fun String.lastChar(): Char = get(this.length - 1)
注意,扩展函数并不允许打破它的封装性。和在类内部定义的方法不同的是,扩展函数不能访问私有的或者是受保护的成员。
导入和扩展函数
对于定义的一个扩展函数,它不会自动地在整个项目范围内生效。相反,如果要使用它,需要进行导入,就像其他任何的类或者函数一样。这是为了避免偶然性的命名冲突。Kotlin 允许用和导入类一样的语法来导入单个的函数:
import strings.lastChar
//星号导入
import strings.*
在 Java 中调用扩展函数
其实,扩展函数是静态函数,它把调用对象作为了它的第一个参数。调用扩展函数,不会创建适配的对象或者任何运行时的额外消耗。
这使得从 Java 中调用 Kotlin 的扩展函数变得非常简单:调用这个静态函数,然后把接收对象作为第一个参数传进去即可。假设它声明在一个叫做 StringUtil.kt 的文件中:
char c = StringUtil.lastChar("Java");
和 Kotlin 版本比较起来,可读性略差。
作为扩展函数的工具函数
现在,可以写一个 joinToString 函数的终极版本了:


