3.8 函数扩展和属性扩展(Extensions)

Kotlin 支持 扩展函数 和 扩展属性。其能够扩展一个类的新功能而无需继承该类或使用像装饰者这样的设计模式等。

大多数时候我们在顶层定义扩展,即直接在包里:

  1. package com.easy.kotlin
  2. val <T> List<T>.lastIndex: Int get() = size - 1
  3. fun String.notEmpty(): Boolean {
  4. return !this.isEmpty()
  5. }

这样我们就可以在整个包里使用这些扩展。

要使用其他包的扩展,我们需要在调用方导入它:

  1. package com.example.usage
  2. import foo.bar.goo // 导入所有名为“goo”的扩展
  3. // 或者
  4. import foo.bar.* // 从“foo.bar”导入一切
  5. fun usage(baz: Baz) {
  6. baz.goo()
  7. }

3.8.1 扩展函数

声明一个扩展函数,我们需要用被扩展的类型来作为前缀。

比如说,我们不喜欢类似下面的双重否定式的逻辑判断(绕脑子):

  1. >>> !"123".isEmpty()
  2. true

我们就可以为String类型扩展一个notEmpty()函数:

  1. >>> fun String.notEmpty():Boolean{
  2. ... return !this.isEmpty()
  3. ... }
  4. >>> "".notEmpty()
  5. false
  6. >>> "123".notEmpty()
  7. true

下面代码为 MutableList<Int> 添加一个swap 函数:

  1. fun MutableList<Int>.swap(index1: Int, index2: Int) {
  2. val tmp = this[index1] // this对应该列表
  3. this[index1] = this[index2]
  4. this[index2] = tmp
  5. }

这个 this 关键字在扩展函数内部对应到接收者对象(传过来的在点.符号前的对象) 现在,我们对任意 MutableList<Int> 调用该函数了。

当然,这个函数对任何 MutableList<T> 起作用,我们可以泛化它:

  1. fun <T> MutableList<T>.mswap(index1: Int, index2: Int) {
  2. val tmp = this[index1] // “this”对应该列表
  3. this[index1] = this[index2]
  4. this[index2] = tmp
  5. }

为了在接收者类型表达式中使用泛型,我们要在函数名前声明泛型参数。

完整代码示例

  1. package com.easy.kotlin
  2. val <T> List<T>.lastIndex: Int get() = size - 1
  3. fun String.notEmpty(): Boolean {
  4. return !this.isEmpty()
  5. }
  6. fun MutableList<Int>.swap(index1: Int, index2: Int) {
  7. val tmp = this[index1] // this对应该列表m
  8. this[index1] = this[index2]
  9. this[index2] = tmp
  10. }
  11. fun <T> MutableList<T>.mswap(index1: Int, index2: Int) {
  12. val tmp = this[index1] // “this”对应该列表
  13. this[index1] = this[index2]
  14. this[index2] = tmp
  15. }
  16. class ExtensionsDemo {
  17. fun useExtensions() {
  18. val a = "abc"
  19. println(a.notEmpty())//true
  20. val mList = mutableListOf<Int>(1, 2, 3, 4, 5)
  21. println("Before Swap:")
  22. println(mList)//[1, 2, 3, 4, 5]
  23. mList.swap(0, mList.size - 1)
  24. println("After Swap:")
  25. println(mList)//[5, 2, 3, 4, 1]
  26. val mmList = mutableListOf<String>("a12", "b34", "c56", "d78")
  27. println("Before Swap:")
  28. println(mmList)//[a12, b34, c56, d78]
  29. mmList.mswap(1, 2)
  30. println("After Swap:")
  31. println(mmList)//[a12, c56, b34, d78]
  32. val mmmList = mutableListOf<Int>(100, 200, 300, 400, 500)
  33. println("Before Swap:")
  34. println(mmmList)
  35. mmmList.mswap(0, mmmList.lastIndex)
  36. println("After Swap:")
  37. println(mmmList)
  38. }
  39. class Inner {
  40. fun useExtensions() {
  41. val mmmList = mutableListOf<Int>(100, 200, 300, 400, 500)
  42. println(mmmList.lastIndex)
  43. }
  44. }
  45. }

测试代码

  1. package com.easy.kotlin
  2. import org.junit.Test
  3. import org.junit.runner.RunWith
  4. import org.junit.runners.JUnit4
  5. @RunWith(JUnit4::class)
  6. class ExtensionsDemoTest {
  7. @Test fun testExtensionsDemo() {
  8. val demo = ExtensionsDemo()
  9. demo.useExtensions()
  10. }
  11. }

扩展不是真正的修改他们所扩展的类。我们定义一个扩展,其实并没有在一个类中插入新函数,仅仅是通过该类型的变量,用点.表达式去调用这个新函数。

3.8.2 扩展属性

和函数类似,Kotlin 支持扩展属性:

  1. val <T> List<T>.lastIndex: Int
  2. get() = size - 1

注意:由于扩展没有实际的将成员插入类中,因此对扩展的属性来说,它的行为只能由显式提供的 getters/setters 定义。

代码示例:

  1. package com.easy.kotlin
  2. val <T> List<T>.lastIndex: Int get() = size - 1

我们可以直接使用包com.easy.kotlin中扩展的属性lastIndex :

Kotlin极简教程