Kotlin中的Lambda和函数式编程

Lambda表达式本质上是一个匿名函数,一段可以被传到其他函数中的代码
语法

左侧为参数区,中间用->分割,右侧为执行区, 外围用大括号包围
Kotlin中的Lambda和函数式编程_第1张图片

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返回集合中第一个满足条件的元素, 如果没有找到满足条件的元素,则返回null

  • groupBy将集合中元素归类,返回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引用,需要用匿名内部类实现

你可能感兴趣的:(Kotlin中的Lambda和函数式编程)