我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Kotlin的Collection与Sequence操作异同点是什么

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Kotlin的Collection与Sequence操作异同点是什么

本文小编为大家详细介绍“Kotlin的Collection与Sequence操作异同点是什么”,内容详细,步骤清晰,细节处理妥当,希望这篇“Kotlin的Collection与Sequence操作异同点是什么”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。

    Collection 的常见操作

    Collection 集合,Kotlin的集合类型和Java不一样,Kotlin的集合分为可变(读写)和不可变(只读)类型(lists, sets, maps, etc),可变类型是在不可变类型前面加Mutable,以我们常用的三种集合类型为例:

    List<out E> - MutableList<E>Set<out E> - MutableSet<E>Map<K, out V> - MutableMap<K, V>

    其实他们的区别就是List实现了Collection接口,而MutableList实现的是List和MutableCollection接口。而 MutableCollection 接口实现了Collection 接口,并且在里面添加了add和remove等操作方法。

    可变不可变只是为了区分只读和读写的操作,他们的操作符方式都是相同的。

    集合的操作符说起来可就太多了

    累计

    //对所有元素求和list.sum()//将集合中的每一个元素代入lambda表达式,然后对lambda表达式的返回值求和list.sumBy {    it % 2}//在一个初始值的基础上,从第一项到最后一项通过一个函数累计所有的元素list.fold(100) { accumulator, element ->    accumulator + element / 2}//同fold,只是迭代的方向相反list.foldRight(100) { accumulator, element ->    accumulator + element / 2}//同fold,只是accumulator的初始值就是集合的第一个元素,element从第二个元素开始list.reduce { accumulator, element ->    accumulator + element / 2}//同reduce但方向相反:accumulator的初始值就是集合的最后一个元素,element从倒数第二个元素开始往前迭代list.reduceRight { accumulator, element ->    accumulator + element / 2}val list = listOf(1, 2, 3, 4, 5, 6)//只要集合中的任何一个元素满足条件(使得lambda表达式返回true),any函数就返回truelist.any {    it >= 0}//集合中的全部元素都满足条件(使得lambda表达式返回true),all函数才返回truelist.all {    it >= 0}//若集合中没有元素满足条件(使lambda表达式返回true),则none函数返回truelist.none {    it < 0}//count函数的返回值为:集合中满足条件的元素的总数list.count {    it >= 0}

    遍历

    //遍历所有元素list.forEach {    print(it)}//同forEach,只是可以同时拿到元素的索引list.forEachIndexed { index, value ->    println("position $index contains a $value")}showFields.forEach { (key, value) ->

    最大最小

    //返回集合中最大的元素,集合为空(empty)则返回nulllist.max()//返回集合中使得lambda表达式返回值最大的元素,集合为空(empty)则返回nulllist.maxBy { it }//返回集合中最小的元素,集合为空(empty)则返回nulllist.min()//返回集合中使得lambda表达式返回值最小的元素,集合为空(empty)则返回nulllist.minBy { it }

    过滤(去除)

    //返回一个新List,去除集合的前n个元素list.drop(2)//返回一个新List,去除集合的后n个元素list.dropLast(2)//返回一个新List,去除集合中满足条件(lambda返回true)的第一个元素list.dropWhile {    it > 3}//返回一个新List,去除集合中满足条件(lambda返回true)的最后一个元素list.dropLastWhile {    it > 3}//返回一个新List,包含前面的n个元素list.take(2)//返回一个新List,包含最后的n个元素list.takeLast(2)//返回一个新List,仅保留集合中满足条件(lambda返回true)的第一个元素list.takeWhile {    it>3}//返回一个新List,仅保留集合中满足条件(lambda返回true)的最后一个元素list.takeLastWhile {    it>3}//返回一个新List,仅保留集合中满足条件(lambda返回true)的元素,其他的都去掉list.filter {    it > 3}//返回一个新List,仅保留集合中不满足条件的元素,其他的都去掉list.filterNot {    it > 3}//返回一个新List,仅保留集合中的非空元素list.filterNotNull()//返回一个新List,仅保留指定索引处的元素list.slice(listOf(0, 1, 2))

    映射

    //将集合中的每一个元素代入lambda表达式,lambda表达式必须返回一个元素//map的返回值是所有lambda表达式的返回值所组成的新List//例如下面的代码和listOf(2,4,6,8,10,12)将产生相同的Listlist.map {    it * 2}//将集合中的每一个元素代入lambda表达式,lambda表达式必须返回一个集合//而flatMap的返回值是所有lambda表达式返回的集合中的元素所组成的新List//例如下面的代码和listOf(1,2,2,3,3,4,4,5,5,6,6,7)将产生相同的Listlist.flatMap {    listOf(it, it + 1)}//和map一样,只是lambda表达式的参数多了一个indexlist.mapIndexed { index, it ->    index * it}//和map一样,只不过只有lambda表达式的非空返回值才会被包含在新List中list.mapNotNull {    it * 2}//根据lambda表达式对集合元素进行分组,返回一个Map//lambda表达式的返回值就是map中元素的key//例如下面的代码和mapOf("even" to listOf(2,4,6),"odd" to listOf(1,3,5))将产生相同的maplist.groupBy {    if (it % 2 == 0) "even" else "odd"}

    元素

    list.contains(2)list.elementAt(0)//返回指定索引处的元素,若索引越界,则返回nulllist.elementAtOrNull(10)//返回指定索引处的元素,若索引越界,则返回lambda表达式的返回值list.elementAtOrElse(10) { index ->    index * 2}//返回list的第一个元素list.first()//返回list中满足条件的第一个元素list.first {    it > 1}//返回list的第一个元素,list为empty则返回nulllist.firstOrNull()//返回list中满足条件的第一个元素,没有满足条件的则返回nulllist.firstOrNull {    it > 1}list.last()list.last { it > 1 }list.lastOrNull()list.lastOrNull { it > 1 }//返回元素2第一次出现在list中的索引,若不存在则返回-1list.indexOf(2)//返回元素2最后一次出现在list中的索引,若不存在则返回-1list.lastIndexOf(2)//返回满足条件的第一个元素的索引list.indexOfFirst {    it > 2}//返回满足条件的最后一个元素的索引list.indexOfLast {    it > 2}//返回满足条件的唯一元素,如果没有满足条件的元素或满足条件的元素多于一个,则抛出异常list.single {    it == 5}//返回满足条件的唯一元素,如果没有满足条件的元素或满足条件的元素多于一个,则返回nulllist.singleOrNull {    it == 5}

    排序&逆序

    val list = listOf(1, 2, 3, 4, 5, 6)//返回一个颠倒元素顺序的新集合list.reversed()list.sorted()//将每个元素代入lambda表达式,根据lambda表达式返回值的大小来对集合进行排序list.sortedBy {    it*2}list.sortedDescending()list.sortedByDescending {    it*2}personList.sortWith(compareBy({ it.age }, { it.name })) val c1: Comparator<Person> = Comparator { o1, o2 ->       if (o2.age == o1.age) {             o1.name.compareTo(o2.name)      } else {           o2.age - o1.age       } }personList.sortWith(c1)  //上面的自定义方式可以通过JavaBean实现Comparable 接口实现自定义的排序    data class Person(var name: String, var age: Int) : Comparable<Person> {     override fun compareTo(other: Person): Int {         if (this.age == other.age) {               return this.name.compareTo(other.name)          } else {             return other.age - this.age          }      }   } //sorted 方法返回排序好的list(已有有排序规则的用sorted,不要用sortedby了) val sorted = personList.sorted()

    Sequence 的常见操作

    Sequence 是 Kotlin 中一个新的概念,用来表示一个延迟计算的集合。Sequence 只存储操作过程,并不处理任何元素,直到遇到终端操作符才开始处理元素,我们也可以通过 asSequence 扩展函数,将现有的集合转换为 Sequence ,代码如下所示

        val list = mutableListOf<Person>()    for (i in 1..10000) {        list.add(Person("name$i", (0..100).random()))    }    list.asSequence()

    当我们拿到结果之后我们还能通过toList再转换为集合。

      list.asSequence().toList()

    Sequence的操作符绝大部分都是和 Collection 类似的。常用的一些操作符是可以直接平替使用的。

      val list2 = list.asSequence()    .filter {        it.age > 50    }.map {        it.name    }.take(3).toList()

    居然他们的操作符都长的一样,效果也都一样,导致 Sequence 与 Collection 就很类似,那么既生瑜何生亮!为什么需要这么个东西?既然 Collection 能实现效果为什么还需要 Sequence 呢?他们的区别又是什么呢?

    区别与对比

    Collection 是立即执行的,每一次中间操作都会立即执行,并且把执行的结果存储到一个容器中,没多一个中间操作符就多一个容器存储结果。

    public inline fun <T, R> Iterable<T>.map(transform: (T) -> R): List<R> {  return mapTo(ArrayList<R>(collectionSizeOrDefault(10)), transform)}public inline fun <T> Iterable<T>.filter(predicate: (T) -> Boolean): List<T> {    return filterTo(ArrayList<T>(), predicate)}

    比如常用的 map 和 filter 都是会新建一个 ArrayList 去存储结果,

    Sequence 是延迟执行的,它有两种类型,中间操作和末端操作 ,主要的区别是中间操作不会立即执行,它们只是被存储起来,中间操作符会返回另一个Sequence,仅当末端操作被调用时,才会按照顺序在每个元素上执行中间操作,然后执行末端操作。

    public fun <T, R> Sequence<T>.map(transform: (T) -> R): Sequence<R> {    return TransformingSequence(this, transform)}public fun <T> Sequence<T>.filter(predicate: (T) -> Boolean): Sequence<T> {    return FilteringSequence(this, true, predicate)}

    比如常用的 map 和 filter 都是直接返回 Sequence 的this 对象。

    public inline fun <T> Sequence<T>.first(predicate: (T) -> Boolean): T {    for (element in this) if (predicate(element)) return element    throw NoSuchElementException("Sequence contains no element matching the predicate.")}

    然后在末端操作中,会对 Sequence 中的元素进行遍历,直到预置条件匹配为止。

    这里我们举一个示例来演示一下:

    我们使用同样的筛选与转换,来看看效果

            val list = mutableListOf<Person>()        for (i in 1..10000) {            list.add(Person("name$i", (0..100).random()))        }        val time = measureTimeMillis {            val list1 = list.filter {                it.age > 50            }.map {                it.name            }.take(3)            YYLogUtils.w("list1$list1")        }        YYLogUtils.w("耗费的时间$time")        val time2 = measureTimeMillis {            val list2 = list.asSequence()                .filter {                    it.age > 50                }.map {                    it.name                }.take(3).toList()            YYLogUtils.w("list2$list2")        }        YYLogUtils.w("耗费的时间2$time2")

    运行结果:

    当集合数量为10000的时候,执行时间能优秀百分之50左右:

    Kotlin的Collection与Sequence操作异同点是什么

    当集合数量为5000的时候,执行时间相差比较接近:

    Kotlin的Collection与Sequence操作异同点是什么

    当集合数量为3000的时候,此时的结果就反过来了,Sequence延时执行的优化效果就不如List转换Sequence再转换List了:

    Kotlin的Collection与Sequence操作异同点是什么

    读到这里,这篇“Kotlin的Collection与Sequence操作异同点是什么”文章已经介绍完毕,想要掌握这篇文章的知识点还需要大家自己动手实践使用过才能领会,如果想了解更多相关内容的文章,欢迎关注编程网行业资讯频道。

    免责声明:

    ① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

    ② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

    Kotlin的Collection与Sequence操作异同点是什么

    下载Word文档到电脑,方便收藏和打印~

    下载Word文档

    猜你喜欢

    Kotlin的Collection与Sequence操作异同点是什么

    本文小编为大家详细介绍“Kotlin的Collection与Sequence操作异同点是什么”,内容详细,步骤清晰,细节处理妥当,希望这篇“Kotlin的Collection与Sequence操作异同点是什么”文章能帮助大家解决疑惑,下面跟
    2023-07-04

    Kotlin的Collection与Sequence操作异同点详解

    这篇文章主要介绍了Kotlin的Collection与Sequence操作异同点详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-11-13

    进程与线程异同点是什么

    这篇文章主要讲解了“进程与线程异同点是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“进程与线程异同点是什么”吧!关于这个问题,有的同学可能已经背得滚瓜烂熟了:“进程是操作系统分配资源的单
    2023-06-27

    vue中el-autocomplete与el-select的异同点是什么

    这篇文章主要介绍“vue中el-autocomplete与el-select的异同点是什么”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“vue中el-autocomplete与el-select的异
    2023-06-30

    link和@import的异同点是什么

    这篇文章主要讲解了“link和@import的异同点是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“link和@import的异同点是什么”吧!页面中使用CSS的方式主要有3种:行内添加
    2023-06-08

    Typescript中interface与type的相同点与不同点是什么

    今天小编给大家分享一下Typescript中interface与type的相同点与不同点是什么的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一
    2023-07-04

    Go语言与Golang的异同是什么?

    Go和Golang是同一种编程语言,它们并没有实质性的区别。"Go"是该编程语言的正式名称,在官方文档和社区中使用广泛。而"Golang"则是Go语言在搜索引擎中的常用关键词,有时人们会在搜索时使用这个名称。Go语言是由Google开发的
    Go语言与Golang的异同是什么?
    2024-02-25

    java同步与异步的区别是什么?

    概念:1、同步:所有的操作都做完,才返回给用户。这样用户在线等待的时间太长,给用户一种卡死了的感觉(就是系统迁移中,点击了迁移,界面就不动了,但是程序还在执行,卡死了的感觉)。这种情况下,用户不能关闭界面,如果关闭了,即迁移程序就中断了。2、异步:将用户请求放
    java同步与异步的区别是什么?
    2015-05-13

    PHP同步与异步的区别是什么

    这篇文章主要介绍“PHP同步与异步的区别是什么”,在日常操作中,相信很多人在PHP同步与异步的区别是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”PHP同步与异步的区别是什么”的疑惑有所帮助!接下来,请跟
    2023-06-25

    Android中同步与异步的关系是什么

    本篇文章为大家展示了Android中同步与异步的关系是什么,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。android 只有UI线程可以刷新界面,其他副线程不行,这样就需要副线程通过通信消息修改刷新
    2023-06-19

    python的30个操作难点分别是是什么

    python的30个操作难点分别是是什么,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。人生苦短,我用 Python,不知道从什么时候开始,这句话开始流行。多年来,Pytho
    2023-06-05

    编程热搜

    • Python 学习之路 - Python
      一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
      Python 学习之路 - Python
    • chatgpt的中文全称是什么
      chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
      chatgpt的中文全称是什么
    • C/C++中extern函数使用详解
    • C/C++可变参数的使用
      可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
      C/C++可变参数的使用
    • css样式文件该放在哪里
    • php中数组下标必须是连续的吗
    • Python 3 教程
      Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
      Python 3 教程
    • Python pip包管理
      一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
      Python pip包管理
    • ubuntu如何重新编译内核
    • 改善Java代码之慎用java动态编译

    目录