文章目录- I . 函数类型
- II . 带参数名的参数列表
- III . 可空函数类型
- IV . 复杂函数类型解读
- V . 函数类型别名
- VI . 带 接收者类型 的函数类型
- VII . 函数类型实例化
- VIII . 函数类型自动推断
- IX . 带接收者的函数类型 与 不带接收者的函数类型 之间的转换
- X . 函数类型变量调用
I . 函数类型
函数类型格式 : 圆括号中定义 参数类型列表 , 使用 -> 由参数列表指向返回值类型 , 表示接受 参数类型列表 中的参数 , 返回 返回值类型 的返回值 ;
( 参数类型列表 ) -> 返回值类型
① 参数列表 : 参数类型的列表 , 多个参数类型使用逗号隔开 ;
② -> 符号 : 由参数列表指向返回值类型 , 该符号用于分割参数类型列表 与 返回值类型 ;
③ 返回值类型 : 只有一个返回值类型 ;
④ 示例 : (Int, Int)->Int
参数类型 , 表示该参数传入两个 Int 类型的参数 , 返回 Int 类型返回值 ;
II . 带参数名的参数列表
1 . 函数类型参数名称 : 参数列表中可以只是参数类型 , 也可以加上参数的变量名称 , 参数名称可以用于说明参数的含义 , 增加函数类型的理解性 ;
2 . 只有参数类型的函数类型 : 参数列表中只有参数类型 ;
( 参数类型1 , 参数类型2 , … 参数类型n ) -> 返回值类型
3 . 有参数名称的函数类型 : 参数列表中每个元素都由 参数名称 : 参数类型 组成 , 多个列表元素使用逗号隔开 ;
( 参数名称1 : 参数类型1 , 参数名称2 : 参数类型2 , … 参数名称n : 参数类型n ) -> 返回值类型
4 . 带参数名称的函数类型示例 :
① 没有参数名的函数类型 : (Int , String)->String
;
② 有参数名的函数类型 : (age : Int , name : String)->String
;
III . 可空函数类型
1 . 函数类型根据是否可空分类 : 函数类型 与 普通变量类型一样 , 也分为可空类型 , 非空类型 两类 ;
2 . 默认非空类型 : 默认的函数类型都是非空类型 , 即函数定义时 , 需要对其进行初始化 , 或延迟初始化 ;
3 . 可空类型函数表示方法 : 可空的函数类型声明时 , 需要在函数类型外部加上括号 , 并在右侧添加 ? 可空符号 ;
( ( 参数列表 ) -> 返回值类型 ) ?
4 . 可空函数类型 与 非空函数类型 示例 :
① 非空函数类型 : (Int , String)->String
, 普通的函数类型 , 默认就是非空的函数类型 ;
② 可空函数类型 : ( (Int , String)->String ) ?
, 在非空函数类型基础上 , 外层添加括号 , 右侧添加 ? , 该函数类型就变成了可空函数类型 ;
IV . 复杂函数类型解读
1 . 复杂函数类型 :
① 参数返回值是普通类型 : 如果函数的参数类型和返回值类型都是普通的类型还好 , 解读起来不是很困难 ;
② 参数返回值是函数类型 : 如果函数类型的参数类型或返回值类型中有函数类型 , 即嵌套的函数类型 , 这些函数类型很复杂 ;
2 . 右箭头 -> 符号的结合性 : 该符号是右结合的 , 解读时 , 先将第一个 -> 符号右边的值理解成返回值 , 再逐步解析 :
( Int , String ) -> ( Float , Double ) -> String
① 分析外层函数类型 ( 函数类型 ) : 首先根据右结合性 , 从左到右开始解读 , 读取到第一个 -> 符号 , 说明这是一个函数类型 , ( Int , String ) 是参数列表 , ( Float , Double ) -> String 是返回值类型 ;
② 分析内层函数类型 ( 外层函数的返回值类型 ) : 然后继续分析 ( Float , Double ) -> String 类型 , 外层的函数类型的 返回值类型 是一个 函数类型 , 该函数类型参数列表是 ( Float , Double ) , 返回值是 String 类型 ;
③ 总结 : 这是函数类型的嵌套 , 外层函数类型的参数类型是 ( Int , String ) , 函数的返回值类型是 ( Float , Double ) -> String 是一个函数类型 ;
3 . 最佳实践 : 复杂的函数类型可读性很差 , 建议使用圆括号注明函数类型的结合性 , 层次 ;
( Int , String ) -> ( Float , Double ) -> String
这样表示虽然没有错误 , 也能读懂 , 但是该表示并不是一目了然 , 读懂理解也要花点时间 , 完全没有必要在代码中出脑筋急转弯题目 , 推荐加上圆括号 , 写成以下形式 :
( Int , String ) -> ( ( Float , Double ) -> String )
这样就可以一目了然 , ( Int , String ) 是参数列表 , ( ( Float , Double ) -> String ) 是返回值类型 , 该返回值类型是一个函数类型 ;
不推荐这种反人类的类型定义 , 建议还是以简洁为主 , 参考 C 语言的函数类型嵌套
int (*p) (int*, int (*f)(int*))
, C 语言的函数 指针类型嵌套int (*(*p2)[5])(int*)
, 这个需要根据复杂指针解读技巧慢慢解读 , 要花费很长时间才能搞懂 ;
V . 函数类型别名
使用 typealias 为函数类型声明一个别名 : 使用函数类型别名 , 能有效降低代码的复杂度 , 提高可读性 , 函数类型别名声明格式如下 :
typealias 别名 = 函数类型
代码示例 : 下面的示例中 , 为 (Int, Int) -> Int 函数类型声明了一个别名 , 该别名与函数类型的作用是一样的 ;
代码语言:javascript复制// 1 . 声明函数类型
typealias mathAdd = (Int, Int) -> Int
// 2 . 为函数类型数理化
var add : mathAdd = {a : Int, b : Int -> a b}
// 3 . 为函数类型实例化
var add2 : (Int, Int) -> Int = {a : Int, b : Int -> a b}
fun main() {
// 4 . 调用函数
var result = add(1,2)
println("$result")
// 5 . 调用函数
var result2 = add2(1,2)
println("$result2")
}
VI . 带 接收者类型 的函数类型
1 . 带 接收者类型 的函数类型 : 函数类型 可以指定 接收者类型 , 格式如下 :
接收者类型.( 参数类型列表 ) -> 返回值类型
① 接收者类型 : 接收者类型 的含义是 , 这个接收者类型对象是 函数类型 实例的接收者 , 该对象拥有该函数 ;
② 上述的 函数类型 定义 表示 : 接收者类型 对象 , 调用 ( 参数类型列表 ) -> 返回值类型 函数类型的函数 , 传入 ( 参数类型列表 ) 类型的参数 , 将返回 “返回值类型” 的返回值 ;
③ 本质 : 实例化该 带接收者的函数类型 变量时 , 相当于为该接收者类型定义了一个扩展函数 ;
2 . 带接收者类型的函数类型示例 :
① 带接收者类型的函数类型 : String.(Int, Float)->String
;
② 示例类型解析 : 在 String 类型对象上 , 调用 (Int, Int)->String
类型的函数 , 在该函数中按照顺序传入 Int , Float 两个类型的参数 , 那么得到一个 String 类型的返回值 ;
VII . 函数类型实例化
函数类型 变量实例化 : 给 函数类型变量 进行赋值 , 可以赋值的类型有以下几种情况 ;
1 . Lambda 表达式 : 可以将 Lambda 表达式赋值给函数类型变量 ;
代码语言:javascript复制// 将 Lambda 表达式赋值给函数类型变量
var add1 : (Int, Int) -> Int = {a : Int, b : Int -> a b}
2 . 匿名函数 : 匿名函数 可以直接赋值给 函数类型 变量 ;
代码语言:javascript复制// 将 匿名函数 赋值给函数类型变量
var add2 : (Int, Int) -> Int = fun (a : Int, b : Int) : Int {return a b}
3 . 已声明的函数 : 已经声明的函数 , 可以直接赋值给函数类型变量 ; 这些函数可以是 顶层函数 , 成员函数 , 局部函数 , 扩展函数 ;
代码语言:javascript复制fun add(a : Int, b : Int) : Int {
return a b
}
// 将顶层的 add 函数赋值给 add3 函数类型变量 :: 用于获取顶层定义的函数
// 如果获取 类中定义的函数 , 可以使用 类名::函数名 获取
var add3 : (Int, Int) -> Int = ::add
4 . 函数变量 : 已经声明的函数类型属性 , 可以是顶层属性 , 成员属性 , 扩展属性 ;
代码语言:javascript复制// 将 Lambda 表达式赋值给函数类型变量
var add1 : (Int, Int) -> Int = {a : Int, b : Int -> a b}
// 将已经定义好的函数类型变量重新赋值给 另外一个函数类型明亮
var add4 : (Int, Int) -> Int = add1
5 . 函数类型 派生类 : 函数类型可以看做一个接口 , 类可以实现该接口 , 在实现类中实现具体的函数操作 , 该 函数类型接口的实现类 , 可以赋值给函数类型变量 ;
代码语言:javascript复制class AddOperation : (Int, Int) -> Int{
override fun invoke(p1: Int, p2: Int): Int {
return p1 p2
}
}
// 将 函数类型 接口派生类对象赋值给 函数类型变量
var add5 : (Int, Int) -> Int = AddOperation()
VIII . 函数类型自动推断
1 . 变量类型推断 : Kotlin 中的变量类型可以不用显示声明 , 可以根据其赋值的类型进行智能类型推断 ;
2 . 函数变量类型推断 : 函数类型变量也具有智能类型推断的性质 ;
var add = {a : Int, b : Int -> a b}
上面的代码中省略了函数类型变量的函数类型 , 其赋值的 Lambda 表达式类型是 (Int, Int) -> Int 类型的 , 因此推断出 add 变量的函数类型是 (Int, Int) -> Int 类型的 ;
IX . 带接收者的函数类型 与 不带接收者的函数类型 之间的转换
带接收者的函数类型 , 可以转换为 不带接收者的函数类型 , 转换规则是 , 带接收者的函数类型的接收者 , 可以转换为不带接收者类型的第一个参数 ;
下面的两个函数类型是等价的 :
① 自带接收者的函数类型 : String.( Int ) -> String
② 不带接收者的函数类型 : ( String, Int ) -> String
带接收者函数类型 与 不带接收者函数类型 转换代码示例 :
代码语言:javascript复制// 字符串 "abc" 调用该函数 , 传入 2 参数 , 结果是 "abcabc"
var fun1 : String.( Int ) -> String = {times : Int -> this.repeat(times)}
// 这是将 接收者 设置成了第一个参数 , 将 String.( Int ) -> String 函数类型的变量 fun1 赋值给了
// ( String, Int ) -> String 函数类型变量
var fun2 : ( String, Int ) -> String = fun1
fun main() {
//"Tom".fun1(2) = TomTom
println(""Tom".fun1(2) = ${"Tom".fun1(2)}")
//fun2("Jerry", 2) = JerryJerry
println("fun2("Jerry", 2) = ${fun2("Jerry", 2)}")
}
① 自带接收者的函数类型 : fun1 是带接收者的 函数类型 变量 , 其类型是 String.( Int ) -> String 类型 ;
② 不带接收者的函数类型 : fun2 是不带接收者的 函数类型变量 , 其类型是 ( String, Int ) -> String 类型 ;
③ 互相赋值 : 将 fun1 变量赋值给 fun2 变量 , 赋值成功 , 说明这两个变量类型是相同的 ;
④ 调用函数 : 分别调用 fun1 和 fun2 函数 , 调用结果相同 ;
执行结果 :
代码语言:javascript复制"Tom".fun1(2) = TomTom
fun2("Jerry", 2) = JerryJerry
X . 函数类型变量调用
函数类型变量调用 :
① invoke 调用 : 可以通过 函数类型变量名.invoke(参数列表) 调用该函数 ;
② 直接调用 : 也可以通过 函数类型变量名(参数列表) 直接调用该函数 , 将该变量名称当做函数名称来使用 ;