Lambda表达式本质上是一个匿名函数,一段可以被传到其他函数中的代码
语法
左侧为参数区,中间用->分割,右侧为执行区, 外围用大括号包围
Kotlin中可以将lambda表达式赋给一个变量,然后将这个变量看待成一个函数使用
val sum = { a: Int, b: Int -> a + b }
println(sum(1,2))
几种语法格式
data class(val name: String, val age: Int)
val users = listOf(User("张三", 32), User("李四", 30))
格式一:
users.maxBy({ u: User -> u.age })
格式二: 如果lambda表达式是一个函数的最后一个参数,可以将表达式移出到括号外
users.maxBy() { u: User -> u.age})
格式三:如果函数只有一个lambda表达式参数,可以省略括号
users.maxBy { u: User -> u.age }
格式四:对于Kotlin编译器能够推断的类型,类型声明也可以省略。
users.maxBy { u -> u.age }
注意:
编译器不能推断出所有变量类型,此处的可以遵循如下原则先不加类型,对于无法通过编译的时候再加。
格式五:当lambda表达式只有一个参数且参数类型可推断时,编译器会自动生成一个名字为it
的默认参数
users.maxBy { it.age }
可见性
Lambda表达式中可以访问外部的局部非final
变量
集合中的函数式API
data class User(val name: String, val age: Int)
val users = listOf(User("张三", 32), User("李四", 30))
-
filter
函数过滤一个集合的元素
val list = listOf(1,2,3,4)
users.filter {it.age > 30}
[User('张三', 32)]
-
map
用定义的行为改变集合中元素
val list = listOf(1,2,3,4)
list.map { it * it }
[1,4,9,16]
如果只想打印对象的某个属性而不是整个对象,也可以应用map
println(users.map {it.name})
[张三, 李四]
-
all
应用lambda判断是否集合中所有元素满足lambda表达式判断条件,只有所有都满足才返回true
val younger30 = { user: User -> user.age <= 30 }
users.all(younger30)
false
-
any
集合中任何元素满足表达式即返回true
users.any(younger30)
true
-
count
统计集合中满足条件的元素个数
users.count(younger30)
1
find
返回集合中第一个满足条件的元素, 如果没有找到满足条件的元素,则返回nullgroupBy
将集合中元素归类,返回map
val users = listOf(User("张三", 32), User("李四", 30), User("王五", 30))
users.groupBy { it.age}
{32=[User(name=张三, age=32)], 29=[User(name=李四, age=30), User(name=王五, age=30)]}
-
asSequence
应用于大集合的链式操作时
当有链式函数调用时,Kotlin会生成中间集合,当集合中存在大量数据时,中间集合会存在低效问题。
users.asSequence()
.map(User::name)
.filter { it.startsWith("张")}
.toList()
注意:
序列调用中顺序是序列化的,即第一个元素执行完整个链然后第二个元素执行,通过终止操作返回。
如上例,User("张三", 32)执行`map`和`filter`然后User("李四", 30)执行`map`和`filter`...
如果是非序列化,则集合中所有元素执行`map`生成一个中间集合,然后这个中间集合中的所有元素
再执行`filter`
函数式接口
当一个接口只有一个抽象方法时,我们称之为函数式接口或者SAM(Single Abstract Method)接口。Java或Android中类似的接口有View.OnClickListener, Runnable等
Kotlin中可以用lambda使调用更加简洁:
button.OnClickListener({view -> ...})
注意:
lambda表达式内部没有自身的引用this,因为编译器是把lambda当做一段代码而不是一个对象,如果
需要this引用,需要用匿名内部类实现