在Kotlin中,函数是一等公民(first class),这意味着函数可以被存储在变量或者数据结构中,它是有类型的。Kotlin使用函数类型来描述一个函数的具体类型。一个完整语法的函数类型如下: 其特点如下: 接收者类型在参数前,和参数所在的小括号用点连接。关于带接收者的函数类型在之后会详细讲解。 既然函数有类型,那函数类型变量的值是什么呢?这就涉及到函数类型的实例化。函数类型的实例化包括以下几种常用方式。 a. lambda表达式: b. 匿名函数: a. 顶级,本地,成员或者扩展函数。例如:String::toInt。 b. 顶级,成员,或者扩展属性:List::size。 c. 构造函数:::Regex。 函数类型的值可以通过invoke操作符调用,以下是示例。 函数字面值是指不声明,而直接作为表达式传递的函数。它包括两种,lambda表达式及匿名函数。 lambda表达式的全语法形式如下: 其特点如下: 其返回值是x-y Kotlin有一个约定:如果函数的最后一个参数是函数,那么作为相应参数传入的lambda表达式可以放在圆括号之外。示例如下: 以lambda表达式作为参数的高阶函数如下: 其调用示例可以简写为: 如果lambda表达式是唯一的参数,那其圆括号可以省略,示例如下: 由于本lambda表达式只有一个参数,其可以隐式声明为it,并去掉该参数的声明和->。示例如下: 自Kotlin1.1起,如果lambda表达式的参数未使用,那么可以用下划线取代其名称: lambda表达式不能显示的指定返回值的类型。如果需要显示指定返回值的类型,则需要使用匿名函数。匿名函数和普通函数非常类似,除了其函数名被省略。其示例如下: 匿名函数的参数和返回值类型的声明与常规函数相同,同时类型如果可以推断出来则可以省略。例如以下示例: anonymousTest中对匿名函数的调用是完整语法形态,由于参数类型和返回值类型可以推断,因此,其可以简写为 请注意,匿名函数参数总是在括号内传递。 允许将函数留在圆括号外的简写语法仅适用于lambda表达式。 lambda表达式与匿名函数之间的另一个区别是非局部返回的行为。一个不带标签的return语句总是在用fun关键字声明的函数中返回。这意味着lambda表达式中的return将从包含它的函数返回,而匿名函数中的return将从匿名函数自身返回。1. 函数类型
(x:Int, y:Int) -> Int
(Int, Int) -> Int
Int.(Int) -> Int
2. 函数类型实例化
{ i, j-> this + i + j }
fun(s1: Int, s2: Int): Int {
return 1
}
class IntTransformer: (Int) -> Int {
override operator fun invoke(x: Int): Int = TODO()
}
val intFunction: (Int) -> Int = IntTransformer()
val stringPlus: (String, String) -> String = String::plus
println(stringPlus.invoke("<-", "->"))
println(stringPlus("Hello, ", "world!"))
3. 函数字面值
3.1 lambda表达式
val sum: (Int, Int) -> Int = {x: Int, y: Int -> x + y}
val sum: (Int, Int) -> Int = {x, y -> x + y}
val sum: (Int, Int) -> Int = {x, y ->
x + y
x-y
}
fun lambdaWrapper(labmda: (Int) -> Int) {
labmda.invoke(1)
}
fun lambdaTest() {
lambdaWrapper() { it ->
it+1
}
}
fun lambdaTest() {
lambdaWrapper{ it ->
it+1
}
}
fun lambdaTest() {
lambdaWrapper{
it+1
}
}
map.forEach { _, value -> println("$value!") }
3.2 匿名函数
fun(x: Int, y: Int): Int {
return x + y
}
fun anonymousWrapper(labmda: (Int) -> Int) {
labmda.invoke(1)
}
fun anonymousTest() {
anonymousWrapper(fun (i: Int): Int {
return i + 1
})
}
fun anonymousTest() {
anonymousWrapper(fun (i) = i+1)
}