我在 LinkedHashMap 中遇到了一些令人困惑的行为
chalice 2.0.3。在 grails 控制台中运行以下脚本:

def m = ["smart-1":[stuff:'asdf']]
println m.getClass()

def p = [id:1]
println m."smart-$p.id"
println m["smart-$p.id"]
println m.get("smart-$p.id")

println m.'smart-1'
println m['smart-1']
println m.get('smart-1')

给出输出:
class java.util.LinkedHashMap
[stuff:asdf]
[stuff:asdf]
null
[stuff:asdf]
[stuff:asdf]
[stuff:asdf]

在集成测试中,我看到了相反的行为 - 我只能
使用 m.get(GStringImpl) 获取 HashMap 的内容(而不是m.get(String) )。

这种行为是预期的还是已知的?

最佳答案

第一:不要在哈希映射键中使用GString。曾经。您几乎总是在检索项目时遇到问题,因为 GString is not a String(该页面上的红色框)并且没有相同的哈希值。相反,请使用以下选项之一:

def key = 'key'
['key': value]
[(key): value]
[("some $key".toString()): value]

这可确保您在使用字符串时始终获得结果。 (因此,对于查找,也始终使用字符串。)

我不确定您为什么会看到这种奇怪的行为,但我不确定,但我有100%的猜测。 get() 方法是一种 Java 方法,而数组样式(可能还有属性样式)的查找是使用 getAt() 实现的,它是一种 Groovy (GDK) 方法。我的猜测是 Groovy 方法知道 GStrings,并且默默地处理转换以确保您不会被绊倒。

最简单的解决方案是始终使用getAt()而不是get:
def m = ['smart-1':[stuff:'asdf']]
println m.getClass()

def p = [id:1]
println m."smart-$p.id"
println m["smart-$p.id"]
println m.getAt("smart-$p.id")

println m.'smart-1'
println m['smart-1']
println m.getAt('smart-1')

哪个工作正常。

更好的解决方案是确保在查找值时使用字符串,如下所示:
println m.get("smart-$p.id".toString())

这也有效。我比较喜欢这个方法,因为直接调用这个方法的时候,你的key是String就更清楚了。在使用数组样式或属性样式访问器时,我仍然会使用普通的 GString,因为这是标准的 Groovy 语法。



这很可能是因为您的哈希图中的键是 GString。

如果 GString 没有任何变量,Groovy 编译器会默默地将其转换为 String 字面量(更好的性能),这就是为什么上面的示例实际上使用 String 作为键,但查找使用的是 GString。

例如
"Hello $name" -> GString('Hello $name')
"Hello Bob"   -> 'Hello Bob'

最后一个想法:只要您使用 groovy,就不要使用 get() ,因为 Groovy 提供了更清晰的 [] 和属性语法。

10-06 09:10