集合的函数式API是用来入门Lambda编程的最佳示例,我们先用kotlin创建一个包含许多水果的集合,然后将水果一个个添加到集合中
val list = ArrayList<String>()
list.add("苹果")
list.add("香蕉")
list.add("梨子")
list.add("葡萄")
这种初始化集合的方式比较麻烦,kotlin为我们提供了一个内置的listOf()函数来简化集合的初始化
val list = listOf("苹果", "香蕉", "梨子", "葡萄")
// 集合也可以用for-in来遍历
for (fruit in list) {
println(fruit)
}
// 这里需要注意,listOf()函数创建的是一个不可变的集合,我们无法对它进行添加,修改,删除操作。
要创建一个可变的集合,kotlin为我们提供了mutableListOf()函数
val list = mutableListOf("苹果", "香蕉", "梨子", "葡萄")
list.add("草莓")
list.add("芒果")
kotlin中Set集合的用法与List几乎一模一样,提供了setOf与mutableSetOf()函数,Set集合中不可以存放重复的元素,如果存放了多个相同的元素,只会保留其中一份。
kotlin中使用Map集合如下
val map = HashMap<String, Int>()
map["apple"] = 1
map["banana"] = 2
// kotlin还提供了一堆mapOf()和mutableMapOf函数简化Map的用法
val map2 = mapOf("apple" to 1, "banana" to 2)
// 我们可以使用for-in遍历map集合
for ((fruit, number) in map2) {
}
用直白的语言来阐述Lambda的话,Lambda就是一小段可以作为参数传递的代码,普通情况下,我们给函数传参时只能传入变量,而使用Lambda却可以传入一小段代码,下面是Lambda表达式的结构
{ 参数1: 参数1类型,参数2:参数2类型 -> 函数体 }
Lambda表达式最外层是一对大括号,如果有参数传入到Lambda表达式中,我们需要声明参数列表,参数列表结尾使用 -> 符号,表示参数列表的结束以及函数体的开始,函数体中可以编写任意行数的代码,最后一行代码会自动作为Lambda表达式的返回值。
我们先来看一个maxBy函数,它接收一个Lambda类型的参数,并且会遍历集合的同时将每次遍历的值作为参数传递给Lambda表达式。maxBy函数的工作原理是根据我们传入的条件来遍历集合,从而找到该条件下的最大值,比如我们想要找到名称最长的水果,那么条件自然是名称的长度
val list = listOf("apple", "banana", "orange", "pear")
val lambda = { fruit: String -> fruit.length }
val maxLengthFruit = list.maxBy(lambda)
这种写法虽然能正常运行,但还是比较啰嗦,我们可以直接将lambda表达式传入maxBy函数中
val list = listOf("apple", "banana", "orange", "pear")
val maxLengthFruit = list.maxBy({fruit: String -> fruit.length})
// 然后kotlin规定,当Lambda参数是函数的最后一个参数时,可以将Lambda表达式移到函数括号外面
val maxLengthFruit = list.maxBy(){fruit: String -> fruit.length}
// kotlin又规定,当Lambda参数是函数唯一一个参数时,可以省略函数的括号
val maxLengthFruit = list {fruit: String -> fruit.length}
// 别慌,我们还可以继续简化,由于kotlin有出色的类型推导机制,Lambda表达式中的参数列表其实在大多数情况下不必声明参数类型
val maxLengthFruit = list {fruit -> fruit.length}
// 最最后,我们还可以简化它,当Lambda表达式的参数列表只有一个参数时,也不必声明参数,而是可以使用it关键字来代替
val maxLengthFruit = list { it.length }
通过上面一步步的简化相信大家对Lambda表达式有一些了解了,我们再来学学集合中比较常用的其他函数式API
// map函数用于将集合中的每个元素都映射成一个另外的值,
// 映射的规则在Lambda表达式中指定,最终生成一个新的集合,
// 比如我们希望集合中的水果名称都变成大写
val newList = list.map { it.toUpperCase() }
// filter函数用来过滤集合中的数据,比如我们只想保留名称在5个字母以内的水果
val newList = list.map { it.length <= 5 }
// any函数用于判断集合中是否至少存在一个元素满足条件,返回true或false
val anyRes = list.any { it.length <= 5 } // 检测是否至少有一个水果名称在五个字符以内
// all函数用于判断集合中是否所有元素都满足条件,返回true或false
val allRes = list.all { it.length <= 5 } // 检测是否所有水果名称都在五个字符以内