遇到 Kotlin elvis 运算符的一些奇怪行为,当尝试从映射中通过键获取某个值时,没有一个 block 不被执行。
map["Key"]?.let {
println("Run block, inserting into existing list")
it.add("value")
} ?: {
println("Elvis block, creating list with value")
map["Key"] = mutableListOf("value")
}
我期望,如果 Key 存在,它将执行 let block ,打印行并将值插入到现有列表中。如果 key 不存在,它将进入 elvis block ,打印行,并插入带有值的新列表。 但事实并非如此。正如 not let block 正在执行,而不是 elvis。
为了解决我的任务,我使用:
map.getOrPut("Key") { mutableListOf() }.add("Value")
问题是,为什么在这种情况下我不能只使用 elvis 和 let 运算符?
Kotlin 版本 1.8.22
最佳答案
放在 ?:
右侧的表达式是:
{
println("Elvis block, creating list with value")
map["Key"] = mutableListOf("value")
}
这只是一个 () -> Unit
类型的 lambda。计算此表达式只会给出 () -> Unit
类型的值。 block 中的代码未执行。
如果您想执行 lambda 中的代码,则应该使用 run
:
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")
}
也就是说,如果您想根据 key 是否存在打印不同的消息,像这样简单的东西更具可读性:
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")
}
当然,如果你不需要打印不同的消息,惯用的方法是getOrPut
。
关于Kotlin elvis 运算符不适用于 map.get(),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/77171506/