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

标签 kotlin dictionary jvm elvis-operator

遇到 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/

相关文章:

java - 无法使用 eclipse 开始调试 java 应用程序

reflection - Kotlin 绑定(bind)的可调用引用不一致

android - 当我发送消息时,聊天屏幕不会更新

java - 从 Swagger 模型中排除实体

Python——追加到字典中

c++ - unordered_map cbegin() + number//常数复杂度?

java - 如何在 JVM 崩溃时创建线程和核心转储

selenium - 无法开始新 session 。响应代码 500。消息 : session not created: This version of ChromeDriver only supports Chrome version 105

为 map 对象执行但不更改值的 C++ 函数

Java堆在jvm内存内部还是外部?