【Kotlin】扩展函数 ① ( 扩展函数简介 | 为 Any 超类定义扩展函数 | private 私有扩展函数 | 泛型扩展函数 | 标准函数 let 函数是泛型扩展函数 )





一、扩展函数简介



现有类 定义 扩展函数 , 可以在 不修改 原有类 的情况下 增加类的功能 ;

Kotlin 中如果类 没有被 open 关键字修饰 , 则该类 不能被继承 , 如果想要扩展该类 , 可以使用 扩展函数 ;

扩展函数 可以作用于 自定义的类 , 也可以作用于 系统自带的类 , 如 String , List , 等 标准库 API 类 ;


某个类 定义扩展函数 , 格式为 :

fun 类名.扩展函数名(参数列表): 返回值类型 {
    
    
	函数体
}

定义扩展函数 与 定义普通函数唯一的区别是 扩展函数 前多了 类名. ;


下面的代码中 , 为 String 定义扩展函数 , 拼接原字符串和扩展函数参数 , 并将结果返回 ;


代码示例 :

/**
 * 为 String 定义扩展函数, 拼接原字符串和扩展函数参数, 并将结果返回
 */
fun String.addStr(str: String): String {
    
    
    println("this = $this, string = $str")
    return this + str
}

fun main() {
    
    
    println("123".addStr("abc"))
}

执行结果 :

this = 123, string = abc
123abc




二、为 Any 超类定义扩展函数



扩展函数 的特点 是 为 父类定义扩展函数 , 子类也可以调用该扩展函数 ;

扫描二维码关注公众号,回复: 14554583 查看本文章

Any 超类 定义 扩展函数 , 那么 所有的 Any 子类 都可以 调用该 扩展函数 ;


一旦在 Any 超类中定义了扩展函数 , 则在整个项目中 , 该扩展函数都有效 ;


这就导致了 Kotlin 的框架非常灵活 , 使用别人的 SDK 时会发现 为各种现有类定义的 扩展函数 ;


代码示例 : 在下面的代码中 , 为 Any 超类 定义了 printSelf 扩展函数 , 所有的类 如 : String , Int , Boolean 等类的实例对象 , 都可以调用该 printSelf 扩展函数 ;

fun Any.printSelf() {
    
    
    println(this)
}

fun main() {
    
    
    "abc".printSelf()
    88.printSelf()
    true.printSelf()
}

执行结果 :

abc
88
true




三、private 私有扩展函数



如果 扩展函数 使用 private 修饰 , 则该扩展函数 只在该 Kotlin 代码文件中有效 , 在其它的 Kotlin 代码中就无法调用了 ;

代码示例 :

在 Hello.kt 代码中 , 使用 private 定义了 Any 超类的扩展函数 , 在本代码中的 main 函数中调用该扩展函数是有效的 ;

private fun Any.printSelf() {
    
    
    println(this)
}

fun main() {
    
    
    "abc".printSelf()
    88.printSelf()
    true.printSelf()
}

在这里插入图片描述

另外一个 Kotlin 代码中 , 调用该 Any 扩展函数 , 就会报编译时报错信息 ;

Cannot access 'printSelf': it is private in file

在这里插入图片描述





四、泛型扩展函数



泛型扩展函数 可以支持 任何类型的 接收者 ( 调用函数的实例对象 ) , 同时还可以 获取 接收者 的 泛型参数类型 ;


泛型扩展函数 格式 : 一般 泛型扩展函数 都是为了 配合 链式编程 , 其 返回值类型 就是 接收者泛型参数类型 ;

fun <T> T.函数名(参数列表): T {
    
    
	函数体
}

代码示例 : 在该代码中 , 定义了一个 泛型扩展函数 , 为泛型 T 定义了扩展函数 printSelf , 在函数中打印 接收者 , 并将接收者返回 , 该泛型可以是任意类型 ;

在 main 函数中 , 先调用 字符串实力对象 的 printSelf 泛型扩展函数打印自身 , 然后调用 字符串的扩展函数 String.addStr 拼接字符串 , 最后再次 调用 字符串实力对象 的 printSelf 泛型扩展函数打印自身 , 这样实现了一个链式编程 ;

// 泛型扩展函数 
fun <T> T.printSelf(): T{
    
    
    println(this)
    return this
}

// 字符串扩展函数
fun String.addStr(str: String): String {
    
    
    println("this = $this, string = $str")
    return this + str
}

fun main() {
    
    
	// 链式编程
    "abc".printSelf().addStr("123").printSelf()
}

执行结果 :

abc
this = abc, string = 123
abc123




五、标准函数 let 函数是泛型扩展函数



标准库 中的 let 函数 , 就是 泛型扩展函数 ,

  • inline 关键字表明该函数是 内联函数 , 其中的 匿名函数 参数在编译时直接将函数体拷贝到使用位置 , 避免创建匿名函数相关对象 , 造成堆内存开销 ;
  • 该函数中涉及到 两个泛型 T 和 R , 在 fun 关键字后声明 ,
  • 泛型 T 定义了一个扩展函数 let ,
  • 传入 (T) -> R 类型的匿名函数 , 该 Lambda 表达式 返回 R 类型 实例对象 ,
  • 该 扩展函数 最终返回 R 类型 实例对象 ,
/**
 * 调用以' this '值作为参数的指定函数[block],并返回其结果。
 *
 * 有关详细使用信息,请参阅[scope functions]的文档
 * (https://kotlinlang.org/docs/reference/scope-functions.html#let)。
 */
@kotlin.internal.InlineOnly
public inline fun <T, R> T.let(block: (T) -> R): R {
    
    
    contract {
    
    
        callsInPlace(block, InvocationKind.EXACTLY_ONCE)
    }
    return block(this)
}

在使用 Lambda 表达式作为参数的时候 , Kotlin 编译器直接将 inline 内联函数函数体 直接拷贝到 使用位置 ; 内联函数 类似于 C 语言中的 预编译指令 宏定义 , 在编译时直接替换拷贝宏定义内容 ; Kotlin 中的 内联函数 也是一种 编译时 进行 宏替换的操作 ;

内联函数参考 【Kotlin】函数 ⑦ ( 内联函数 | Lambda 表达式弊端 | “ 内联 “ 机制避免内存开销 - 将使用 Lambda 表达式作为参数的函数定义为内联函数 | 内联函数本质 - 宏替换 ) 博客进行理解 ;


Kotlin 中的标准库函数 , 参考 【Kotlin】标准库函数总结 ( apply 函数 | let 函数 | run 函数 | with 函数 | also 函数 | takeIf 函数 | takeUnless 函数 ) , 基本都是 泛型扩展函数 ;

猜你喜欢

转载自blog.csdn.net/han1202012/article/details/128750675