Kotlin elvis 运算符不适用于 map.get()

Ant*_*ony 2 dictionary jvm kotlin elvis-operator

遇到 Kotlin elvis 运算符的一些奇怪行为,当尝试从映射中通过键获取某个值时,并且没有一个块不被执行。

map["Key"]?.let {
    println("Run block, inserting into existing list")
    it.add("value")
} ?: {
    println("Elvis block, creating list with value")
    map["Key"] = mutableListOf("value")
}
Run Code Online (Sandbox Code Playgroud)

我期望,如果 Key 存在,它将执行 let 块,打印行并将值插入到现有列表中。如果 key 不存在,它将进入 elvis 块,打印行,并插入带有值的新列表。但事实并非如此。不要让块被执行,而不是 elvis。

为了解决我的任务,我使用:

map.getOrPut("Key") { mutableListOf() }.add("Value")
Run Code Online (Sandbox Code Playgroud)

问题是,为什么在这种情况下我不能只使用 elvis 和 let 运算符?

Kotlin 版本 1.8.22

Swe*_*per 5

您放在右侧的表达式?:是:

{
    println("Elvis block, creating list with value")
    map["Key"] = mutableListOf("value")
}
Run Code Online (Sandbox Code Playgroud)

这只是一个类型为 的 lambda () -> Unit。计算这个表达式只是给你一个类型的值() -> Unit。块中的代码不会被执行。

run如果你想执行 lambda 中的代码,你应该使用:

map["Key"]?.let {
    println("Run block, inserting into existing list")
    it.add("value")
} ?: run {
    println("Elvis block, creating list with value")
    map["Key"] = mutableListOf("value")
}
Run Code Online (Sandbox Code Playgroud)

也就是说,如果您想根据密钥是否存在打印不同的消息,像这样简单的东西更具可读性:

val list = map["Key"]
if (list != null) {
    println("Run block, inserting into existing list")
    list.add("value")
} else {
    println("Elvis block, creating list with value")
    map["Key"] = mutableListOf("value")
}
Run Code Online (Sandbox Code Playgroud)

当然,如果你不需要打印不同的消息,惯用的方式是getOrPut