【Kotlin 集合概述】可变参数vararg、中缀函数infix以及解构声明(二十)

导读大纲

    • 1.1 使用集合: vararg、infix 调用和解构声明
      • 1.1.1 扩展 Java 集合 API
      • 1.1.2 vararg: 接受任意数量参数的函数
      • 1.1.3 处理pairs: Infix 调用和解构声明

1.1 使用集合: vararg、infix 调用和解构声明

  1. 本节将介绍 Kotlin 标准库中用于处理集合的一些函数

  2. 同时,还介绍一些相关的语言特性(主要涉及以下三大特性)

    • vararg 关键字允许你声明一个包含任意数量参数的函数
    • infix 中缀函数, 可让您在调用某些单参数函数时更加简单
    • 解构声明(Destructuring declarations)可将单个复合值解包为多个变量

1.1.1 扩展 Java 集合 API

  1. 我们认为 Kotlin 中的集合是与 Java 中相同的类,但具有扩展的 API
    • 比如,这里的获取列表中最后一个元素计算数字集合之和
fun main() {val strings: List<String> = listOf("first", "second", "fourteenth")strings.last()// fourteenthval numbers: Collection<Int> = setOf(1, 14, 2)numbers.sum()// 17
}
  1. 为什么在 Kotlin 中,尽管集合是 Java 库类的实例,却可以用集合做很多事情
    • 现在,答案应该很清楚: last 和 sum 函数被声明为扩展函数
      1. 关于扩展函数和扩展属性以及顶级函数与顶级属性前面已经讲过,这里不再赘述
        • 顶级函数和顶级属性
        • 扩展函数系列
      2. 学习扩展函数时,我曾说过,定义的扩展函数必须导入到当前文件才会生效
        • 这里无需导入是因为集合相关的扩展函数总是默认导入到你的 Kotlin 文件
    • last函数是 List 类的扩展,这里的T是泛型的类型参数,可以先不用理会
      1. 这个扩展的实现很简单,应该都能看懂
      2. 比如在扩展函数中调用this.isEmpty(),this指代的是List的实例对象,可以省略
        title
    • sum函数只针对Iterable类型,简单说就是只有元素是Int类型的迭代对象可以调用该方法
      title
public fun <T> List<T>.last(): T {if (isEmpty())throw NoSuchElementException("List is empty.")return this[lastIndex]
}
@kotlin.jvm.JvmName("sumOfInt")
public fun Iterable<Int>.sum(): Int {var sum: Int = 0for (element in this) {sum += element}return sum
}

1.1.2 vararg: 接受任意数量参数的函数

  1. 调用函数创建列表时,可以传递任意数量的参数
    • 如果在标准库中查找该函数的声明方式,会发现其签名如下
      1. fun listOf(vararg values: T): List { /* implementation */ }
        title
val list = listOf(235711)
  1. 这种方法利用一种语言特性: vararg
    • 将任意数量的值打包到数组然后传递给方法
    • <1> Kotlin 的 vararg 与 Java 中的可变参数类似,但语法略有不同
      1. Kotlin 在参数上使用 vararg 修饰符
      2. 而 Java 是在类型后面加上三个点
// 在Kotlin中
fun main() {val strings: List<String> = listOf("first", "second", "fourteenth")  // <1>strings.last()
}
// 在Java中
public class Example {public static void Test (String  ...args) {                         // <1>System.out.println(Arrays.toString(args));}public static void main(String[] args){Test("1", "2", "3");}
}
  1. Kotlin 和 Java 的另一个不同之处在于
    • 需要传递的参数已经打包在数组中时,调用函数的语法
    • 在Java中,可以原封不动地传递数组, 而Kotlin则要求显式地解包数组
      1. 以便每个数组元素都成为被调用函数的单独参数
      2. wtf,这不就是Python的*args,对, 没错
    • <1> 这一功能被称为展开运算符,使用它直接在相应参数前加上"*"字符
      1. 这里"展开"args数组(其中包含传递给main函数的命令行参数)
        • 将其用作 listOf 函数的可变参数
    • <2> 展开运算符可将数组中的值与一些其他值组合起来
      1. Java 中不支持这种操作,这就是最大的不同点
fun main(args: Array<String>) {val list = listOf("args: ", *args)       // <1>println(list)val extraArgs = listOf("hello", *args)   // <2>println(extraArgs)
}

1.1.3 处理pairs: Infix 调用和解构声明

  1. 之前学习集合的简单创建时, 讲过可以使用 mapOf 函数
    • <1> to 并不是内置结构,而是一种特殊的方法调用,即 infix 调用
      1. 如果小伙伴记性不错,应该记得之前for循环迭代整数时使用过"100 downTo 1"语法
      2. 没戳,infix fun Int.downTo(to: Int) 也是一个中缀函数,即使用infix修饰符的函数
        title
val map = mapOf(1 to "one", 7 to "seven", 53 to "fifty-three")  // <1>
  1. 在 infix 调用
    • 方法名紧接在目标对象名和参数之间, 没有额外的分隔符
    • <1> 以下两个调用是等价的
      1. 第一个是按常规方式调用函数
      2. 第二个使用infix简洁语法来调用函数
fun main(args: Array<String>) {println(1.to("one"))                // <1>println(1 to "one")                 // <1>
}
=============================
(1, one)
(1, one)
  1. 对于只有一个所需参数普通方法和扩展函数,可以使用 infix 调用
    • 要使用中缀简洁语法调用一个函数,需要用 infix 修饰符对其进行标记
    • <1> to 函数返回 Pair 的实例,Pair 是 Kotlin 标准库中的一个类
      1. Pair表示一对元素
        title
infix fun Any.to(other: Any) = Pair(this, other)   // <1>
  1. 请注意,您可以直接用一个 Pair 对象初始化两个变量
    • <1> 这一功能称为解构
      title
fun main(args: Array<String>) {val (number, name) = 1 to "one"    // <1>println("$number to $name")      
}
  1. 解构功能并不局限于Pair对象
    • 例如, 也可以用 map entry来初始化 key 和 value 这两个变量
    • <1> 这也适用于循环,正如在实现joinToString时使用的 withIndex 函数
      1. 这里解构声明的作用就在于不用额外定义一个 index 变量
        • 通过迭代时自增来跟踪当前元素的索引值
      2. 解构出来的index本身就代表当前元素的索引
for ((index, element) in collection.withIndex()) {   // <1>println("$index: $element")
}
  1. to 函数是一个扩展函数, 可以创建一对任意元素
    • 这意味着它是通用接收器的扩展:
      1. 可以编写 1 to “one”, “one” to 1, list to list.size(),以此类推
    • <1> 看看 mapOf 函数的实际签名, 与 listOf 类似
      1. mapOf也接受可变参数,但这次参数是键值对(Pair<K, V>)
    • 尽管在 Kotlin 中创建一个新的 map 看起来像是一个特殊的结构
      1. 但它却是一个infix语法的常规函数,Kotlin的简洁性也算是初露锋芒
// <1>
public fun <K, V> mapOf(vararg pairs: Pair<K, V>): Map<K, V> =if (pairs.size > 0) pairs.toMap(LinkedHashMap(mapCapacity(pairs.size))) else emptyMap()

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.xdnf.cn/news/1547239.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

Vue3+Element-UI Plus登录静态页

<script setup> import {reactive, ref} from "vue";const formRefref() const formModelreactive({username:,password: }) const formRulesreactive({username:[{required:true,message:请输入账号,trigger:blur}],password:[{required:true,message:请输入密…

阳光精机第一大客户双刃剑效应:关联交易引关注,产能利用率忽高忽低

《港湾商业观察》杨丹妮 8月27日&#xff0c;北交所向无锡阳光精机股份有限公司&#xff08;以下简称&#xff1a;阳光精机&#xff09;下发第二轮审核问询函&#xff0c;此前阳光精机于2023年12月29日递表北交所。 阳光精机公司主要生产精密主轴、主辊、弧形导轨等机床功能部…

【数据结构】AVL树相关知识详细梳理

1. AVL树的概念 AVL的全称是Adelson-Velsky-Landis&#xff0c;其名称来源于其发明者Adelson、Velsky和Landis&#xff0c; 是平衡二叉树搜索树。 它的出现是由于二叉搜索树虽可以缩短查找的效率&#xff0c;但如果数据有序或接近有序二叉搜索树将退化为单支树&#xff0c;查找…

常见的TTL,RS232,RS485,IIC,SPI,UART之间的联系和区别

简单总结 图片来源 RS232,RS485可参考&#xff0c;IIC&#xff0c;SPI,UART可参考 烧录程序中常听到的一句话就是USB转TTL&#xff0c;但严格来说算是USB传输数据的协议转换成TTL&#xff08;Transistor-Transistor Logic&#xff09;协议传输数据。首先&#xff0c;usb是常见…

15、网络安全合规由来与要素

数据来源&#xff1a;1.网络安全合规由来与要素_哔哩哔哩_bilibili 合规由来 合规&#xff08;Compliance&#xff09;&#xff1a;指服从、顺从和遵从的行为&#xff0c;强调使公司的经营活动与法律、监管及内部规则保持一致。合规涉及公司适应法律法规及社会规范等规则的经营…

[附源码]在线音乐系统+SpringBoot+Vue前后端分离

今天带来一款优秀的项目&#xff1a;在线音乐系统源码 。 系统采用的流行的前后端分离结构&#xff0c;内含功能包括 "管理后台"&#xff0c;“用户端”&#xff0c;“评论系统”&#xff0c;“歌手&#xff0c;歌曲管理”&#xff0c;“用户系统”,"统计"…

bugku-头等舱

根据题目提示&#xff0c;查看请求头试一下&#xff0c; 得到flag&#xff0c;直接提交

【JavaScript】LeetCode:56-60

文章目录 56 路径总和Ⅲ57 二叉树的最近公共祖先58 二叉树中的最大路径59 岛屿数量60 腐烂的橘子 56 路径总和Ⅲ 递归遍历每个节点所有可能的路径。pathSum()&#xff1a;返回所有节点满足条件的路径数目&#xff0c;traversal()&#xff1a;返回当前遍历节点满足条件的路径数目…

CloudMusic:免费听歌

本文所涉及所有资源均在 传知代码平台可获取。 目录 概述 演示效果 视频演示 图片展示 核心逻辑 获取歌曲图片 提取搜索结果 使用方式 部署方式 Docker部署1 构建镜像 Web站点部署2 附件下载 概述 CloudMusic是一款全网歌曲免费听的web项目&#xff0c;无需任何数据库&#x…

19、网络安全合规复盘

数据来源&#xff1a;5.网络安全合规复盘_哔哩哔哩_bilibili

【Java】异常处理 —— Throwable 及其应用

通过一张图来展示Throwable类的继承体系&#xff0c;如图2所示。 图2 Throwable异常体系结构图 ● Error类称为错误类&#xff0c;它表示Java运行时产生的系统内部错误或资源耗尽的错误&#xff0c;是比较严重的&#xff0c;仅靠修改程序本身是不能恢复执行的&#xff0c;例如…

3D建模软件 | Blender v4.2.2 绿色版

Blender是一款功能强大的免费开源3D创作套件&#xff0c;适用于创建3D可视化效果&#xff0c;如静态图像、3D动画、视觉特效以及视频编辑。Blender以其跨平台兼容性、高效内存管理、统一的工作流程和活跃的社区支持而受到独立艺术家和小型工作室的青睐。 它提供了从建模、渲染…

Ubuntu下安装向日葵:闪退

下载 https://sunlogin.oray.com/download 初次安装 $ sudo dpkg -i SunloginClient_15.2.0.63064_amd64.deb 正在选中未选择的软件包 sunloginclient。 (正在读取数据库 ... 系统当前共安装有 234281 个文件和目录。) 准备解压 SunloginClient_15.2.0.63064_amd64.deb ..…

使用bat命令在没有java的环境下启动jar包

使用bat命令在没有java的环境下启动jar包 先看一下目录下面的文件 里面有三个比较重要的文件 clean.bat&#xff1a;用于清除占用程序的端口 一键启动_x64.bat&#xff1a;用于启动全部的项目 jre8_win64&#xff1a;用于jar所需要的java环境 注意事项&#xff1a; 关于jar…

MySQL 中优化 COUNT()查询的实用指南

在 MySQL 数据库的使用中&#xff0c;我们经常会用到 COUNT()函数来统计行数或满足特定条件的行数。然而&#xff0c;在处理大规模数据时&#xff0c;COUNT()查询可能会变得非常缓慢&#xff0c;影响数据库的性能。那么&#xff0c;如何在 MySQL 中优化 COUNT()查询呢&#xff…

Redis一些简单通用命令和认识常用数据类型和编码方式

通用命令 get() / set() 这是Redis中两个最为核心的命令。 set插入 这里的key 和 value都是字符串&#xff0c;我们可以加双引号 或者单引号&#xff0c;或者不加。 get查找 如果查询的key值不存在&#xff0c;那么会返回一个 nil &#xff0c;也就是代表空 在Redis中命令…

【C++位图】构建灵活的空间效率工具

目录 位图位图的基本概念如何用位图表示数据位图的基本操作setresettest 封装位图的设计 总结 在计算机科学中&#xff0c;位图&#xff08;Bitmap&#xff09;是一种高效的空间管理数据结构&#xff0c;广泛应用于各种场景&#xff0c;如集合操作、图像处理和资源管理。与传统…

什么是开放式耳机?具有什么特色?非常值得入手的蓝牙耳机推荐

开放式耳机是当下较为热门的一种耳机类型。它具有以下特点&#xff1a; 设计结构&#xff1a; 呈现开放式的构造&#xff0c;不会完全堵住耳道。如此一来&#xff0c;外界声音能够较容易地被使用者听到&#xff0c;在使用耳机时可以保持对周围环境的察觉。比如在户外&#xf…

绿色新纪元:光伏技术飞跃与能源体系重塑

近年来&#xff0c;光伏电池技术取得了突破性进展。新型高效光伏材料如钙钛矿、有机光伏等不断涌现&#xff0c;这些材料在转换效率和稳定性上均表现出色&#xff0c;为光伏产业注入了新的活力。同时&#xff0c;光伏组件的智能化、轻量化设计也日益成为趋势&#xff0c;使得光…

Go基础学习06-Golang标准库container/list(双向链表)深入讲解;延迟初始化技术;Element;List;Ring

基础介绍 单向链表中的每个节点包含数据和指向下一个节点的指针。其特点是每个节点只知道下一个节点的位置&#xff0c;使得数据只能单向遍历。 示意图如下&#xff1a; 双向链表中的每个节点都包含指向前一个节点和后一个节点的指针。这使得在双向链表中可以从前向后或从后…