22.Kotlin解构声明详解

Kotlin解构声明详解

有时把一个对象解构成很多变量会很方便,例如:

val (name, age) = person

这种语法称为解构声明 。一个解构声明同时创建多个变量。 我们已经声明了两个新变量: name 和 age,并且可以独立使用它们:

println(name)
println(age)

一个解构声明会被编译成以下代码:

val name = person.component1()
val age = person.component2()

其中的 component1()component2()函数是在 Kotlin 中广泛使用的约定原则例子。 任何表达式都可以出现在解构声明的右侧,只要可以对它调用所需数量的 component 函数即可。 当然,可以有 component3()component4() 等等。

请注意,componentN()函数需要用 operator关键字标记,以允许在解构声明中使用它们。

解构声明也可以用在 for循环中:当你写:

for ((a, b) in collection) { …… }

变量 a 和 b 的值取自对集合中的元素上调用 component1() 和 component2() 的返回值。

例:从函数中返回两个变量
假设我们需要从一个函数返回两个东西。例如,一个结果对象和一个某种状态。 在 Kotlin 中一个简洁的实现方式是声明一个数据类,并返回其实例:

data class Result(val result: Int, val status: Status)
fun function(……): Result {
    // 各种计算

    return Result(result, status)
}

// 现在,使用该函数:
val (result, status) = function(……)

因为数据类自动声明 componentN()函数,所以这里可以用解构声明。

注意:我们也可以使用标准类 Pair并且让 function()返回 Pair<Int, Status>, 但是让数据合理命名通常更好。

例:解构声明和映射

可能遍历一个映射(map)最好的方式就是这样:

for ((key, value) in map) {
    // 使用该 key、value 做些事情
}

为使其能用,我们应该
通过提供一个iterator() 函数将映射表示为一个值的序列;
通过提供函数component1()component2()来将每个元素呈现为一对。

当然事实上,标准库提供了这样的扩展:

operator fun <K, V> Map<K, V>.iterator(): Iterator<Map.Entry<K, V>> = entrySet().iterator()
operator fun <K, V> Map.Entry<K, V>.component1() = getKey()
operator fun <K, V> Map.Entry<K, V>.component2() = getValue()

因此你可以在 for循环中对映射(以及数据类实例的集合等)自由使用解构声明。

下划线用于未使用的变量

如果在解构声明中你不需要某个变量,那么可以用下划线取代其名称:

val (_, status) = getResult()

对于以这种方式跳过的组件,不会调用相应的 componentN() 操作符函数。

在 lambda 表达式中解构

你可以对 lambda表达式参数使用解构声明语法。 如果lambda 表达式具有Pair类型(或者Map.Entry 或任何其他具有相应 componentN函数的类型)的参数,那么可以通过将它们放在括号中来引入多个新参数来取代单个新参数:

map.mapValues { entry -> "${entry.value}!" }
map.mapValues { (key, value) -> "$value!" }

注意声明两个参数和声明一个解构对来取代单个参数之间的区别:

{ a //-> …… } // 一个参数
{ a, b //-> …… } // 两个参数
{ (a, b) //-> …… } // 一个解构对
{ (a, b), c //-> …… } // 一个解构对以及其他参数

如果解构的参数中的一个组件未使用,那么可以将其替换为下划线,以避免编造其名称:

map.mapValues { (_, value) -> "$value!" }

你可以指定整个解构的参数的类型或者分别指定特定组件的类型:

map.mapValues { (_, value): Map.Entry<Int, String> -> "$value!" }

map.mapValues { (_, value: String) -> "$value!" }

完整示例代码如下:

data class MyResult(val result: String, val status: Int)

fun myMethod(): MyResult {
    return MyResult("Success", 1)
}

fun myMethod2(): Pair<String, Int> {
    return Pair("Sucess", 1)
}

fun main(args: Array<String>) {
    val (result, status) = myMethod()

    println(result)
    println(status)

    println("----------")

    val (result2, status2) = myMethod2()
    println(result2)
    println(status2)

    println("----------")

    val map = mapOf<String, String>("a" to "aa", "b" to "bb", "c" to "cc")

    for ((key, value) in map) {
        println("key: $key, value: $value")
    }

    println("------------")

    map.mapValues { entry -> " ${entry.value} hello" }.forEach { println(it) }

    println("------------")


    map.mapValues { (key, value) -> "$value world" }.forEach { println(it) }

    println("------------")

    map.mapValues { (_, value) -> "$value welcome" }.forEach { println(it) }

    println("------------")


    //kotlin允许我们为结构声明整体指定类型,也可以为每一个具体的component指定类型

    map.mapValues { (_, value): Map.Entry<String, String> -> "$value person" }.forEach { println(it) }


    println("-------------")

    map.mapValues { (_, value: String) -> "$value people" }.forEach { println(it) }

}

输出结果

Success
1
----------
Sucess
1
----------
key: a, value: aa
key: b, value: bb
key: c, value: cc
------------
a= aa hello
b= bb hello
c= cc hello
------------
a=aa world
b=bb world
c=cc world
------------
a=aa welcome
b=bb welcome
c=cc welcome
------------
a=aa person
b=bb person
c=cc person
-------------
a=aa people
b=bb people
c=cc people
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,752评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,100评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,244评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,099评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,210评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,307评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,346评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,133评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,546评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,849评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,019评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,702评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,331评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,030评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,260评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,871评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,898评论 2 351

推荐阅读更多精彩内容