Kotlin学习系列——集合的函数式API

几乎各种语言都对集合操作提供了方便的库函数,Kotlin也不例外,今天就来说说Kotlin中与集合操作相关的函数式API。

在开始之前先说一下这里的演示数据,后面演示数据的代码将不再重复出现:

//数字集合
val numList = listOf(1, 2, 3, 4, 5, 6)
//People数据类
data class People(val name: String, val age: Int)
val peopleList = listOf(People("小红", 22),
        People("小明", 23),
        People("小白", 22))

 
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
filter和map过滤变换

flitermap各种集合操作API的基础,如果你了解过Java8,应该对这两个函数不陌生,这里我们就不与Java8对比了,直接介绍他们在Kotlin中的用法。

filter是对一个集合中的所有元素应用一个表达式判断,然后返回一个判断结果为true的元素的新集合。说白了就是从一个集合中筛选出符合条件的元素并返回一个新集合。下面直接给出示例代码:

fun main(args: Array<String>) {
    //筛选出大于等于3的元素
    println(numList.filter { it >= 3 })
    //筛选出年龄等于22岁的人
    println(peopleList.filter { it.age == 22 })
}

 
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7

如果对于代码中的Lambda表达式或者是数据类不了解的小伙伴,可以查看历史文章,这里就不再赘述了。

运行结果如下:

[3, 4, 5, 6]
[People(name=小红, age=22), People(name=小白, age=22)]

 
 
  • 1
  • 2

map是对一个集合中的每个元素应用一个表达式,然后生成一个新的集合,也是非常简单的,直接上代码:

fun main(args: Array<String>) {
    //求集合中每个元素的平方值
    println(numList.map { it * it })
    //提取出出集合中每个People对象的name属性
    println(peopleList.map { it.name })
}

 
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7

运行结果如下:

[1, 4, 9, 16, 25, 36]
[小红, 小明, 小白]

 
 
  • 1
  • 2

当然上面的filtermap也可以链式调用,对一个集合做多次操作,比如我们想筛选出年龄等于22岁的人的姓名那就可以这样写:

fun main(args: Array<String>) {
    println(peopleList.filter { it.age == 22}.map { it.name })
}

 
 
  • 1
  • 2
  • 3

输出结果如下:

[小红, 小白]

 
 
  • 1

filtermap除了对list进行操作外,还可以操作map,注意这里的map是指key-value的map。看一下下面的代码分别对map的key和value进行过滤和变换操作。

fun main(args: Array<String>) {
    val numMap = mapOf("0" to 48, "1" to 49, "2" to 50, "3" to 51, "4" to 52)
    println(numMap.filter { it.key == "1" && it.value == 49 })
    println(numMap.filterKeys { it == "1" })
    println(numMap.filterValues { it == 48 })
    //返回一个list
    println(numMap.map { "key = " + it.key to "value = " + it.value})
    //返回一个新的map
    println(numMap.mapKeys { "key = " + it.key })
    //返回一个新的map
    println(numMap.mapValues { "value = " + it.value })
}

 
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14

输出结果如下:

{1=49}
{1=49}
{0=48}
[(key = 0, value = 48), (key = 1, value = 49), (key = 2, value = 50), (key = 3, value = 51), (key = 4, value = 52)]
{key = 0=48, key = 1=49, key = 2=50, key = 3=51, key = 4=52}
{0=value = 48, 1=value = 49, 2=value = 50, 3=value = 51, 4=value = 52}

 
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
all、any、count和find查找

allanycountfind都是和查找相关的API。

all用来确定一个集合中的元素是否都满足某个条件,而any是判断集合中是否有元素满足某个条件。下面是示例代码:

fun main(args: Array<String>) {
    //判断是否所有元素都小于6
    println(numList.all { it <= 6 })
    //判断是否所有人的年龄都小于25岁
    println(peopleList.all { it.age < 25 })
    //判断是否存在大于100的数
    println(numList.any { it > 10 })
    //判断是否存在年龄大于25岁的人
    println(peopleList.any{ it.age > 25})
}

 
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13

输出结果如下:

true
true
false
false

 
 
  • 1
  • 2
  • 3
  • 4

count用于对我们上面的判断检查进行统计,统计有多少个元素是满足条件的。就像下面这样:

fun main(args: Array<String>) {
    //统计集合中小于等于3的元素的个数
    println(numList.count { it <= 3 })
    //统计集合中年龄小于等于22的人的个数
    println(peopleList.count { it.age <= 22 })
}

 
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7

输出结果如下:

3
2

 
 
  • 1
  • 2

其实上面的需求我们还有一种方式可以实现,就是先过滤出满足条件的元素组成一个新的集合,再获取这个新的集合的大小,就像下面这样:

fun main(args: Array<String>) {
    println(numList.filter { it <= 3 }.size)
    println(peopleList.filter { it.age <= 22 }.size)
}

 
 
  • 1
  • 2
  • 3
  • 4
  • 5

上面的代码虽然可以完成我们的需求,但是它相较于前一种count的方式,生成了一个新的集合,其实我们根本不关心元素本身,所以count方式更加高效。

find用于返回第一个符合条件的元素

groupBy分组

groupBy用于对元素进行分组,分组后的结果是一个Map,它的类型是这样的Map<key, List>,如果你了解SQL,一定对groupBy不陌生。下面看看我们根据年龄对一个集合进行分组的代码示例:

fun main(args: Array<String>) {
    println(peopleList.groupBy { it.age })
}

 
 
  • 1
  • 2
  • 3

输出结果如下:

{22=[People(name=小红, age=22), People(name=小白, age=22)], 23=[People(name=小明, age=23)]}

 
 
  • 1

可以看出它的输出结果是Map<Int, List<People>>类型。

flatMap和flatten铺平

flatMapflatten的都是对集合的操作,你可以把它们简单理解为对集合元素进行展开再合并到一起形成一个新的集合。

其中flatMap是对一个集合进行map 变换 以后,再将变化后的多个集合合并为一个集合。如果你只是想单纯地合并多个集合而不需要变换,那就选用flatten,下面是具体的示例代码:

fun main(args: Array<String>) {
    val strList = listOf("abc", "def", "hij")
    println(strList.flatMap { it.toList() })
    val lists = listOf(listOf("abc", "bnm"), listOf("acv", "wnm"), listOf("awm"))
    println(lists.flatten())
}

 
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7

输出结果如下:

[a, b, c, d, e, f, h, i, j]
[abc, bnm, acv, wnm]

 
 
  • 1
  • 2

是不是发现比我们手动合并多个集合方便多了。



再先看一段代码:


fun main(args: Array<String>) {
        val strList = listOf("abc", "def", "hiej")
        println(strList.flatMap { it.toList() })
        println(strList.map { it.toList() })
        println(strList.map { it.toList() }.flatten())
}

看输出:

[a, b, c, d, e, f, h, i, e, j]
[[a, b, c], [d, e, f], [h, i, e, j]]
[a, b, c, d, e, f, h, i, e, j]

看源码:

public inline fun <T, R> Iterable<T>.map(transform: (T) -> R): List<R> {
    return mapTo(ArrayList<R>(collectionSizeOrDefault(10)), transform)
}
public inline fun <T, R, C : MutableCollection<in R>> Iterable<T>
						.mapTo(destination: C, transform: (T) -> R): C {
    for (item in this)
        destination.add(transform(item))
    return destination
}
// =================
public inline fun <T, R> Iterable<T>.flatMap(transform: (T) -> Iterable<R>): List<R> {
    return flatMapTo(ArrayList<R>(), transform)
}
public inline fun <T, R, C : MutableCollection<in R>> Iterable<T>
			.flatMapTo(destination: C, transform: (T) -> Iterable<R>): C {
    for (element in this) {
        val list = transform(element)
        destination.addAll(list)
    }
    return destination
}

 
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24

对比 map 与 flatMap 发现区别在于一个是调用 add() , 一个是调用 addAll().

这就明白了,为什么上面的输出 flatMap 之后是一个集合,每个元素是原始数据;而 map 之后,每个元素是一个集合。

结论强调kt 的 flatMap 是对集合的操作,是一个平铺操作


写在最后

通过对Kotlin中集合操作API的介绍,又一次领略到Kotlin的简洁,以后遇到集合处理相关的需求,应该首先看看Kotlin是否已经提供了相关的API,然后选用合适的API进行处理即可。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值